C++知识点总结(22):模拟算法

news/2025/2/13 19:09:42/

一、概念

模拟算法

根据题目描述进行筛选提取关键要素,按需求书写代码解决实际问题的算法。

二、步骤

1、提取题目的关键要素

2、根据关键要素的需求完成代码

三、关键要素

1、题目目的

2、样例的执行逻辑(样例分析)

3、数据范围(十年OI一场空 不开long long见祖宗)

四、核弹演习

1. 审题

题目描述

R国最近在秘密进行一项军事模拟演习,其内容为:核弹在城市中爆炸后,哪些位置是能够幸存的。但是这项实验无法在现实中进行,只能通过计算机进行模拟,现任命你为首批研发人员,完成该模拟程序的研发。将军给出了如下的要求:

  1. 将长宽为n,m的地图输入进程序,每个坐标都有自己的防护等级p,最小为0,意味着无防护,最大为20,即地下防空洞。
  2. 输入核弹投放的坐标,以及核弹的当量(辐射强度)。
  3. 核弹的爆炸范围为核弹的辐射范围
  4. 输入任意一个坐标,若其实际辐射等级≥0,则是安全的,输出" Safe",否则输出"D

anger "

坐标的实际辐射等级 =坐标的防护等级 —爆炸产生的辐射等级。

核弹的爆炸范围计算:在坐标(x,y)的位置引爆一颗当量为K的核弹,即此时投弹点的位置辐射强度k,投弹点周围向外距离为1的位置,辐射等级为k-1,以此类推,直到K

=1时,不再向外进行辐射。

例如在一个4x4的地区,所有坐标均无防护,在(3,3)引爆一颗当量为2的核弹,其效果:

引爆前:

引爆后:

0000

0000

0000

0111

0000

0121

0000

0111

输入描述

共n+3行

第1行包含两个整数,为地图的长和宽

第2~n+1行,每行包含m个整数,为每个坐标的防护等级p

第n+2行,包含3个整数,分别为投弹点坐标(x,y)以及核弹当量k

第n+3行,包含2个整数,为任意的一个位置

输出描述

1行,包含一个字符串。

2. 思路

1. 输入一个二维数组

2. 放炸弹(重点)

3. 判断某一处是否安全

放炸弹的过程

观察一下题目中的引爆前和引爆后的矩阵,它就是环形矩阵

复习一下【环形矩阵】:

for (p = 1 ~ n) // 遍历层数
{for (r = p ~ 2n-p) // 遍历行{for (c = p ~ 2n-p){a[i][j]++;}}
}

得出规律:

zx-(k-p) ~ zx+(k-p)

zy - (k-p) ~ zy+(k-p)

3. 参考答案

#include <iostream>
using namespace std;int n, m;
int zx, zy;
int ax, ay;
int k;
int a[1005][1005];int main()
{// 输入cin >> n >> m;for (int i = 1; i <= n; i++){for (int j = 1; j <= m; j++){cin >> a[i][j];}}cin >> zx >> zy >> k;cin >> ax >> ay;// 放炸弹for (int p = 1; p <= k; p++){for (int i = zx-(k-p); i <= zx+(k-p); i++){for (int j = zy-(k-p); j <= zy+(k-p); j++){a[i][j]--;}}}// 判断if (a[ax][ay] >= 0){cout << "Safe";}else{cout << "Danger";}return 0;
}

我们发现一个bug:下标越界的问题。我们想到,如果炸弹的范围超出了地图,那么将会出现下标越界的问题,且会报错。我们可以在三重for的内部这么写:

                if (i >= 1 && i <= n && j >= 1 && j <= m){a[i][j]--;}

如果你比较懒,可以只比较一次,这样想起来更简单:

#include <iostream>
#include <cmath>
using namespace std;int n, m;
int zx, zy;
int ax, ay;
int k;
int a[1005][1005];int main()
{// 输入cin >> n >> m;for (int i = 1; i <= n; i++){for (int j = 1; j <= m; j++){cin >> a[i][j];}}cin >> zx >> zy >> k;cin >> ax >> ay;// 判断if (a[ax][ay] - (max(abs(ax-zx), abs(ay-zy))) > 0) // 防护等级 - 辐射等级{cout << "Safe";}else{cout << "Danger";}return 0;
}

五、热搜榜

1. 审题

模拟并制作一个前10名的热搜榜。

2. 思路

1. 定义一个结构体(id+次数+名字)

2. 用cnt[]统计新热搜的次数,如果为新热搜,放入

3. 判断是否把旧热搜挤掉,如果是则次数清零,右指针移动一格

3. 参考答案

#include <iostream>
#include <algorithm>
using namespace std;int n;
int index, temp;
int l = 1, r = 1;
int cnt[105];struct Node
{int id, name, num;
}a[1005];bool cmp(Node a, Node b)
{if (a.num != b.num){return a.num > b.num;}return a.id < b.id;
}int main()
{cin >> n;for (int i = 1; i <= n; i++){cin >> temp;cnt[temp]++;if (cnt[temp] == 1){a[r].name = temp;a[r].id = i;r++;}if (r - l > 10){cnt[a[l].name] = 0;l++;}}for (int i = l; i <= r-1; i++){a[i].num = cnt[a[i].name];}sort(a+l, a+r, cmp);index = 1;cout << "前10热搜榜\n";for (int i = l; i <= r-1; i++){cout << "Top " << index++ << ": " << a[i].name << endl;}return 0;
}

六、海港

1. 审题

题目描述

某个海港每天都有船只进出,每艘船进港时都会记录下进港的时间和船只的编号。现在需要统计每天进港港口内的各个船只的数量,但是只统计最近24小时内进港的船只数量,超过24小时的船只不计算在内。请你编写程序实现这个统计功能。

输入格式 

第一行输入一个整数n,表示有n条船进港的记录。 接下来n行,每行输入两个整数t和k,分别表示船只进港的时间和船只的编号。其中,t表示进港时间距离当天0点的秒数,k表示船只的编号。

输出格式

对于每一条进港记录,输出最近24小时内进港的船只数量。

样例解释

第一艘船进港时,船只编号1的数量为1。 第二艘船进港时,船只编号1和2的数量为2。 第三艘船进港时,船只编号1、2和3的数量为3。 第四艘船进港时,船只编号1、2和3的数量为3。 第五艘船进港时,船只编号1、2、3和4的数量为4。 第六艘船进港时,船只编号1、2、3和4的数量为4。

2. 思路

1. 输入人,存储Ta的到达时间和国家,放入对应的属性中,国家次数增加1

2. 如果是新国家,不同的国家数量增加1,右指针移动一格

3. 当两个指针的时间差>=86400,遍历左指针移动若干格

3. 参考答案

#include <iostream>
using namespace std;int n,t,k,l=1,r=1,ans,cnt[100005];struct Node
{int ti, ci;
}a[300005];int main()
{cin >> n;for (int i = 1; i <= n; i++){cin >> t >> k;for (int j = 1; j <= k; j++){cin >> a[r].ci;a[r].ti = t;cnt[a[r].ci]++;if (cnt[a[r].ci] == 1){ans++;}r++;}while (a[r-1].ti - a[l].ti >= 86400){cnt[a[l].ci]--;if (cnt[a[l].ci] == 0){ans--;}l++;}cout << ans << endl;}return 0;
}


http://www.ppmy.cn/news/1362292.html

相关文章

C#,数值计算,矩阵的乔莱斯基分解(Cholesky decomposition)算法与源代码

一、安德烈路易斯乔尔斯基 安德烈路易斯乔尔斯基出生于法国波尔多以北的查伦特斯海域的蒙古扬。他在波尔多参加了Lyce e&#xff0c;并于1892年11月14日获得学士学位的第一部分&#xff0c;于1893年7月24日获得第二部分。1895年10月15日&#xff0c;乔尔斯基进入莱科尔理工学院…

最新计算机毕业设计题目100例

文章目录 0 前言1 计算机毕设选题推荐2 开题指导3 最后 0 前言 大家好&#xff01;大四的同学们毕业设计即将开始了&#xff0c;你们做好准备了吗&#xff1f; 学长给大家精心整理了最新的计算机毕业设计选题&#xff0c;希望能为你们提供帮助。如果在选题过程中有任何疑问&a…

【嵌入式学习】QT-Day3-Qt基础

1> 思维导图 https://lingjun.life/wiki/EmbeddedNote/20QT 2> 完善登录界面 完善对话框&#xff0c;点击登录对话框&#xff0c;如果账号和密码匹配&#xff0c;则弹出信息对话框&#xff0c;给出提示”登录成功“&#xff0c;提供一个Ok按钮&#xff0c;用户点击Ok后…

python实现把文件夹内jpg图片替换为png

注意&#xff1a;代码一生成的png文件会和jpg文件共存&#xff0c;想删除所有jpg文件请执行代码二 代码一&#xff1a;将jpg文件转存为png文件 import os from PIL import Image# 设置输入文件夹和输出文件夹 input_folder jpg文件夹路径 # 替换成你的jpg图片所在文件夹路径…

Linux应用-ElasticSearch安装

ElasticSearch安装部署 简介 全文搜索属于最常见的需求&#xff0c;开源的 Elasticsearch &#xff08;以下简称 es&#xff09;是目前全文搜索引擎的首选。 它可以快速地储存、搜索和分析海量数据。维基百科、Stack Overflow、Github 都采用它。 Elasticsearch简称es&…

Ubuntu22部署MySQL5.7详细教程

Ubuntu22部署MySQL5.7详细教程 一、下载MySQL安装包二、安装MySQL三、启动MySQL检查状态登录MySQL 四、开启远程访问功能1、允许其他主机通过root访问数据库2、修改配置文件&#xff0c;允许其他IP通过自定义端口访问 五、使用Navicat连接数据库 默认情况下&#xff0c;Ubuntu2…

Vue中如何使用dayjs

Day.js中文网Day.js是一个极简的JavaScript库&#xff0c;可以为现代浏览器解析、验证、操作和显示日期和时间。https://dayjs.fenxianglu.cn/ 单位不区别大小写&#xff0c;支持复数和缩写形式 单位缩写描述 date D日期 [1,31]dayd星期 [0,6]&#xff08;星期日0&#xff0c…

常用显示屏学习——LCD12864(含高级驱动程序)

LCD12864液晶显示屏 屏幕介绍 ① 可显示四行字符&#xff0c;每行可显示8个汉字或者16个数字和字母&#xff1b; ②可串行通信和并行通信&#xff1b; ③ 串口接口管脚信号 通信方法 &#xff08;一&#xff09;八位并行通信方法 &#xff08;二&#xff09;串行通信方法 用…