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


    一、概念

    模拟算法

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

    二、步骤

    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. 判断某一处是否安全

    放炸弹的过程

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

    复习一下【环形矩阵】:

    1. for (p = 1 ~ n) // 遍历层数
    2. {
    3. for (r = p ~ 2n-p) // 遍历行
    4. {
    5. for (c = p ~ 2n-p)
    6. {
    7. a[i][j]++;
    8. }
    9. }
    10. }

    得出规律:

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

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

    3. 参考答案

    1. #include
    2. using namespace std;
    3. int n, m;
    4. int zx, zy;
    5. int ax, ay;
    6. int k;
    7. int a[1005][1005];
    8. int main()
    9. {
    10. // 输入
    11. cin >> n >> m;
    12. for (int i = 1; i <= n; i++)
    13. {
    14. for (int j = 1; j <= m; j++)
    15. {
    16. cin >> a[i][j];
    17. }
    18. }
    19. cin >> zx >> zy >> k;
    20. cin >> ax >> ay;
    21. // 放炸弹
    22. for (int p = 1; p <= k; p++)
    23. {
    24. for (int i = zx-(k-p); i <= zx+(k-p); i++)
    25. {
    26. for (int j = zy-(k-p); j <= zy+(k-p); j++)
    27. {
    28. a[i][j]--;
    29. }
    30. }
    31. }
    32. // 判断
    33. if (a[ax][ay] >= 0)
    34. {
    35. cout << "Safe";
    36. }
    37. else
    38. {
    39. cout << "Danger";
    40. }
    41. return 0;
    42. }

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

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

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

    1. #include
    2. #include
    3. using namespace std;
    4. int n, m;
    5. int zx, zy;
    6. int ax, ay;
    7. int k;
    8. int a[1005][1005];
    9. int main()
    10. {
    11. // 输入
    12. cin >> n >> m;
    13. for (int i = 1; i <= n; i++)
    14. {
    15. for (int j = 1; j <= m; j++)
    16. {
    17. cin >> a[i][j];
    18. }
    19. }
    20. cin >> zx >> zy >> k;
    21. cin >> ax >> ay;
    22. // 判断
    23. if (a[ax][ay] - (max(abs(ax-zx), abs(ay-zy))) > 0) // 防护等级 - 辐射等级
    24. {
    25. cout << "Safe";
    26. }
    27. else
    28. {
    29. cout << "Danger";
    30. }
    31. return 0;
    32. }

    五、热搜榜

    1. 审题

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

    2. 思路

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

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

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

    3. 参考答案

    1. #include
    2. #include
    3. using namespace std;
    4. int n;
    5. int index, temp;
    6. int l = 1, r = 1;
    7. int cnt[105];
    8. struct Node
    9. {
    10. int id, name, num;
    11. }a[1005];
    12. bool cmp(Node a, Node b)
    13. {
    14. if (a.num != b.num)
    15. {
    16. return a.num > b.num;
    17. }
    18. return a.id < b.id;
    19. }
    20. int main()
    21. {
    22. cin >> n;
    23. for (int i = 1; i <= n; i++)
    24. {
    25. cin >> temp;
    26. cnt[temp]++;
    27. if (cnt[temp] == 1)
    28. {
    29. a[r].name = temp;
    30. a[r].id = i;
    31. r++;
    32. }
    33. if (r - l > 10)
    34. {
    35. cnt[a[l].name] = 0;
    36. l++;
    37. }
    38. }
    39. for (int i = l; i <= r-1; i++)
    40. {
    41. a[i].num = cnt[a[i].name];
    42. }
    43. sort(a+l, a+r, cmp);
    44. index = 1;
    45. cout << "前10热搜榜\n";
    46. for (int i = l; i <= r-1; i++)
    47. {
    48. cout << "Top " << index++ << ": " << a[i].name << endl;
    49. }
    50. return 0;
    51. }

    六、海港

    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. 参考答案

    1. #include
    2. using namespace std;
    3. int n,t,k,l=1,r=1,ans,cnt[100005];
    4. struct Node
    5. {
    6. int ti, ci;
    7. }a[300005];
    8. int main()
    9. {
    10. cin >> n;
    11. for (int i = 1; i <= n; i++)
    12. {
    13. cin >> t >> k;
    14. for (int j = 1; j <= k; j++)
    15. {
    16. cin >> a[r].ci;
    17. a[r].ti = t;
    18. cnt[a[r].ci]++;
    19. if (cnt[a[r].ci] == 1)
    20. {
    21. ans++;
    22. }
    23. r++;
    24. }
    25. while (a[r-1].ti - a[l].ti >= 86400)
    26. {
    27. cnt[a[l].ci]--;
    28. if (cnt[a[l].ci] == 0)
    29. {
    30. ans--;
    31. }
    32. l++;
    33. }
    34. cout << ans << endl;
    35. }
    36. return 0;
    37. }

  • 相关阅读:
    【Java面试】MongoDB
    select...for update到底是加了行锁,还是表锁?
    【LeetCode每日一题合集】2023.9.11-2023.9.17(⭐反悔贪心&拓扑排序&Floyd)
    Python学习(五):字典核心底层原理
    IDEA断点调试技巧
    竞赛trick-AWP对抗训练的即插即用实现
    微服务理论知识
    HTTP协议中GET请求和POST请求的区别
    聊聊RocketMQ Rebalance负载均衡触发机制浅析
    软件工程师:持续反馈
  • 原文地址:https://blog.csdn.net/joe_g12345/article/details/136274262