当前位置:   article > 正文

蓝桥杯2020年第十一届C/C++B组(第一次)省赛习题题解_2020年第十一届蓝桥杯省赛 第一场 b组

2020年第十一届蓝桥杯省赛 第一场 b组

目录

第一题:跑步训练(模拟)

第二题:纪念日(模拟)

第三题:合并检测(数学)

第四题 REPEAT 程序(模拟)

第五题:矩阵(dp)

第六题:整除序列(模拟)

第七题:解码(模拟+string)

第八题:走方格(DFS+dp)

第九题:整数拼接(数学+幂)


题目来源:2020年第十届C/C++ B组第一场蓝桥杯省赛真题_戎码一生的博客-CSDN博客_蓝桥杯c++真题


第一题:跑步训练(模拟)

 核心思路:

由于答案要求的是秒数,所以需要将上述的单位转换为秒,那么就可以得到平均每秒的能量损耗;因为有剩下的能量,不足以支撑1分钟的跑步的情况,所以需要对此情况进行特判

  1. #include<iostream>
  2. using namespace std;
  3. int main()
  4. {
  5. int n = 10000;
  6. int run = 600 / 60;//每秒的消耗
  7. int time = 0;
  8. while (n)
  9. {
  10. if (n - 600 < 0)//如果剩余的能量不支撑于1分钟run,则取整秒break
  11. {
  12. time = time * 60 + n / run;
  13. break;
  14. }
  15. n -= 600;
  16. n += 300;
  17. time += 2;
  18. }
  19. cout << time;
  20. return 0;
  21. }

第二题:纪念日(模拟)

 核心思路:

因为要求的是天数,平年365天,闰年366天,枚举天数的同时,特判平年还是闰年即可

  1. #include<iostream>
  2. using namespace std;
  3. int yearDay(int year)
  4. {
  5. if ((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0))
  6. {
  7. return 366;
  8. }
  9. return 365;
  10. }
  11. int main()
  12. {
  13. int time = 0;
  14. for (int i = 1922; i <= 2020; i++)
  15. {
  16. time += yearDay(i);
  17. }
  18. time -= 22;//23-22=1
  19. cout << time * 24 * 60 << endl;
  20. return 0;
  21. }

第三题:合并检测(数学)

 核心思路:

假设总人数为100人,那么就有,对于k枚举 1~100 的范围内

(1)对于 总人数100%k==0 的情况:需要的总的试剂盒为:(100/i)+ i

········解释:(100/i):100个人中每次分为i个人 为 1组,那么总共就分为了 100/i组

········解释: +i是因为 它有1%的阳性率,(概率呈均匀分布)那么这100个人中就必定有人为阳性,那么就需要对,患阳性的那一组重新进行i个人的检测

(2)对于 总人数100%k!=0 的情况:需要的总的试剂盒为:(100/i)+ 1 + i

········解释:对于除不尽的情况需要进行+1 处理,即被分为了(100/i)+1组,剩下的的解释就与上面相同

  1. #include<iostream>
  2. using namespace std;
  3. int main()
  4. {
  5. int ans = 0;
  6. int sum = 0x3f3f3f;//初始化为一个很大的数,因为要求的是小于的情况
  7. for (int i = 1; i <= 100; i++)//枚举所有的检测数
  8. {
  9. if (100 % i == 0)
  10. {
  11. if (100 / i + i < sum)
  12. {
  13. sum = 100 / i + i;
  14. ans = i;//题目要求的是被分为多少组是最小的,所以需要进行记录
  15. }
  16. }
  17. else
  18. {
  19. if (100 / i + 1 + i < sum)
  20. {
  21. sum = 100 / i + 1 + i;
  22. ans = i;//记录当前最小的检测数
  23. }
  24. }
  25. }
  26. cout << ans << endl;
  27. }

第四题 REPEAT 程序(模拟)

 核心思路:模拟

  1. #include<iostream>
  2. using namespace std;
  3. int main()
  4. {
  5. int ans = 0;
  6. for (int i = 0; i < 2; i++)
  7. {
  8. ans += 4;
  9. for (int j = 0; j < 5; j++)
  10. {
  11. for (int k = 0; k < 6; k++)
  12. {
  13. ans += 5;
  14. }
  15. ans += 7;
  16. }
  17. ans += 8;
  18. }
  19. ans += 9;
  20. cout << ans << endl;
  21. return 0;
  22. }

第五题:矩阵(dp)

 思路来自:  

第十一届蓝桥杯 ——矩阵_六级不考550+不改名-CSDN博客

  1. #include <iostream>
  2. using namespace std;
  3. int f[1020][1020];//含义:第一行中选i个数字,第二行中选择j个数字的方案数
  4. int main()
  5. {
  6. f[0][0] = 1; // 两行一个数字都不放,也是一种方案
  7. for (int i = 0; i <= 1010; i++)
  8. for (int j = 0; j <= i; j++)
  9. {
  10. if (i - 1 >= j) // 转移前的状态也要合法,即第一行的数量不小于第二行的数量
  11. f[i][j] += f[i - 1][j] % 2020;
  12. if (j - 1 >= 0)
  13. f[i][j] += f[i][j - 1] % 2020;
  14. }
  15. cout << f[1010][1010] << endl;//全选的情况
  16. return 0;
  17. }

第六题:整除序列(模拟)

  1. #include<iostream>
  2. #include<cstdio>
  3. using namespace std;
  4. int main()
  5. {
  6. long long n;
  7. scanf("%lld",&n);
  8. while(n)
  9. {
  10. printf("%lld ",n);
  11. n>>=1;
  12. }
  13. return 0;
  14. }

第七题:解码(模拟+string)

 

  1. #include <iostream>
  2. #include <cstring>
  3. using namespace std;
  4. int main()
  5. {
  6. string s;
  7. cin >> s;
  8. string ans;
  9. for (int i = 0; i < s.size(); i ++)
  10. {
  11. if(s[i] >= '0' && s[i] <= '9')
  12. for (int j = 0; j < (s[i] - '0') - 1; j ++) ans += s[i - 1];
  13. else ans += s[i];
  14. }
  15. cout << ans << endl;
  16. return 0;
  17. }

第八题:走方格(DFS+dp)

 

 思路来自:

第十一届蓝桥杯——走方格_六级不考550+不改名-CSDN博客

  1. #include <iostream>
  2. using namespace std;
  3. const int N = 50;
  4. int f[N][N];
  5. int main()
  6. {
  7. int n, m;
  8. cin >> n >> m;
  9. for (int i = 1; i <= n; i ++)
  10. for (int j = 1; j <= m; j ++)
  11. {
  12. if(i % 2 == 0 && j % 2 == 0) continue;
  13. if(i == 1 && j == 1) f[i][j] = 1; // 起点到起点只有一种方案
  14. else if(i == 1) f[i][j] = 1; // 第一行只能从左边走过来
  15. else if(j == 1) f[i][j] = 1; // 第一列只能从上面走下来
  16. else f[i][j] = f[i][j - 1] + f[i - 1][j]; // 其余情况:既能从左边走过来,也可以从上面走下来
  17. }
  18. cout << f[n][m] << endl;
  19. return 0;
  20. }

第九题:整数拼接(数学+幂)

暴力:

  1. #define _CRT_SECURE_NO_WARNINGS
  2. #include<iostream>
  3. #include<cstdio>
  4. #include<algorithm>
  5. #include<cmath>
  6. using namespace std;
  7. const int N = 100010;
  8. int n, k;
  9. int a[N];
  10. bool check(int x,int y)
  11. {
  12. int cnt = 0;
  13. int t = y;
  14. while (y)//123
  15. {
  16. cnt++;
  17. y /= 10;
  18. }
  19. x = x * pow(10, cnt) + t;
  20. if (x % k == 0) return true;
  21. else return false;
  22. }
  23. int main()
  24. {
  25. cin >> n >> k;
  26. for (int i = 0; i < n; i++) scanf("%d", &a[i]);
  27. int ans = 0;
  28. for (int i = 0; i < n; i++)
  29. {
  30. for (int j = 0; j < n; j++)
  31. {
  32. if (check(a[i], a[j]) && i!=j) ans++;
  33. }
  34. }
  35. cout << ans << endl;
  36. return 0;
  37. }

优化:

AcWing 2068. 整数拼接 - AcWing

  1. #include <iostream>
  2. #include <algorithm>
  3. #include <cstring>
  4. using namespace std;
  5. typedef long long ll;
  6. const int N = 100010;
  7. int n, mod; // n 即题目中 n,mod 即上述 k
  8. ll ans; // ans 存答案,由于最多会有 n ^ 2 中情况,所以需要开 ll
  9. int a[N]; // a 存输入数据
  10. int cnt[11][N]; // cnt 即上述数组 cnt
  11. int log_10(int x) // 返回 1 + log10(x)
  12. {
  13. int res = 0;
  14. while (x) x /= 10, res ++ ;
  15. return res;
  16. }
  17. void work() // 从前往后跑一遍
  18. {
  19. for (int i = 0; i < n; i ++ )
  20. {
  21. ans += cnt[log_10(a[i])][(mod - a[i] % mod) % mod]; // 累加 cnt
  22. for (int j = 0, power = 1; j < 11; j ++ ) // 将 a[i] 的 0 ~ 10 次方 % mod 的结果计入 cnt
  23. {
  24. cnt[j][power * 1ll * a[i] % mod] ++ ;
  25. power = power * 10 % mod;
  26. }
  27. }
  28. }
  29. int main()
  30. {
  31. scanf("%d%d", &n, &mod);
  32. for (int i = 0; i < n; i ++ )
  33. scanf("%d", a + i);
  34. work();
  35. memset(cnt, 0, sizeof cnt); // 别忘了在第二次跑之前清空一下 cnt 数组
  36. // 其实并不需要从前往后,从后往前写两遍。
  37. // 只需要将数组反转一下,再从前往后跑一遍即可
  38. reverse(a, a + n);
  39. work();
  40. printf("%lld\n", ans);
  41. return 0;
  42. }
  43. 作者:垫底抽风
  44. 链接:https://www.acwing.com/solution/content/15969/
  45. 来源:AcWing
  46. 著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop】
推荐阅读
相关标签
  

闽ICP备14008679号