> 文档中心 > 【蓝桥备赛冲刺】2021年第十二届省赛B组第一场题解C/C++陆续更新状态ing

【蓝桥备赛冲刺】2021年第十二届省赛B组第一场题解C/C++陆续更新状态ing

目录

A题:空间

B题:卡片

C题:直线

D题:货物摆放

E题:路径


A题:空间

问题描述

题目描述小蓝准备用256MB 的内存空间开一个数组,数组的每个元素都是32 位二进制整数。如果不考虑程序占用的空间和维护内存需要的辅助空间。请问256MB 的空间可以存储多少个32 位二进制整数?

思路分析

知道内存单位转换即可 1M = 1024k  1k = 1024B  1B = 8b。需要注意的是不能直接 *8 / 32,而是要 / 4,否则会 int 型数据溢出。

题目代码

#include using namespace std;int main() {    /* 1M = 1024k  1k = 1024B  1B = 8b */    cout << 256 * 1024 * 1024 / 4;    return 0;}

题目答案

67108864

B题:卡片

问题描述

小蓝有很多数字卡片,每张卡片上都是数字0 到9。小蓝准备用这些卡片来拼一些数,他想从1 开始拼出正整数,每拼一个,就保存起来。卡片就不能用来拼其它数了。小蓝想知道自己能从1 拼到多少。例如,当小蓝有30 张卡片,其中0 到9 各3 张,则小蓝可以拼出1 到10。但是拼11 时卡片1 已经只有一张了,不够拼出11。现在小蓝手里有0 到9 的卡片各2021 张,共20210 张,请问小蓝可以从1拼到多少?提示:建议使用计算机编程解决问题。

思路分析

初始化每张卡片数量为2021,枚举每次用到的卡片并计数,当有任意一张卡片的数量小于0时停止计数。

题目代码

#include using namespace std;int nb[10];bool find(int n) {    while (n) { int N = n % 10; n /= 10; if (--nb[N] < 0)     return false;  // 卡片已经用完    }    return true;  // 卡片有剩余}int main() {    int num = 0;    for (int i = 0; i < 10; i++)  // 卡片初始化 nb[i] = 2021;    for (int i = 1; find(i); i++) num++;    cout << num;    return 0;}

题目答案

3181

C题:直线

在平面直角坐标系中,两点可以确定一条直线。如果有多点在一条直线上,那么这些点中任意两点确定的直线是同一条。给定平面上2 × 3 个整点{(x, y)|0 ≤ x < 2, 0 ≤ y < 3, x ∈ Z, y ∈ Z},即横坐标是0 到1 (包含0 和1) 之间的整数、纵坐标是0 到2 (包含0 和2) 之间的整数的点。这些点一共确定了11 条不同的直线。给定平面上20 × 21 个整点{(x, y)|0 ≤ x < 20, 0 ≤ y < 21, x ∈ Z, y ∈ Z},即横坐标是0 到19 (包含0 和19) 之间的整数、纵坐标是0 到20 (包含0 和20) 之间的整数的点。请问这些点一共确定了多少条不同的直线。

问题描述

在平面直角坐标系中,两点可以确定一条直线。如果有多点在一条直线上,那么这些点中任意两点确定的直线是同一条。给定平面上2 × 3 个整点{(x, y)|0 ≤ x < 2, 0 ≤ y < 3, x ∈ Z, y ∈ Z},即横坐标是0 到1 (包含0 和1) 之间的整数、纵坐标是0 到2 (包含0 和2) 之间的整数的点。这些点一共确定了11 条不同的直线。给定平面上20 × 21 个整点{(x, y)|0 ≤ x < 20, 0 ≤ y < 21, x ∈ Z, y ∈ Z},即横坐标是0 到19 (包含0 和19) 之间的整数、纵坐标是0 到20 (包含0 和20) 之间的整数的点。请问这些点一共确定了多少条不同的直线。

思路分析

枚举所有点对并求出直线方程,当斜率k或者截距b有任意一个不同时直线不同,斜率不存在特殊考虑。

题目代码

#include #include #include using namespace std;const int N = 200000;int n;struct Line {    double k, b;  // 斜率k 截距b} l[N];bool rule(Line A, Line B) {  // 制定sort排序规则    if (A.k != B.k) return A.k < B.k;    return A.b < B.b;}int main() {    /* 枚举所有点对 */    for (int x1 = 0; x1 < 20; x1++) for (int y1 = 0; y1 < 21; y1++)     for (int x2 = 0; x2 < 20; x2++)  for (int y2 = 0; y2 < 21; y2++)      if (x1 != x2) {  // 不考虑斜率不存在   double k = (double)(y2 - y1) / (x2 - x1);   double b = y1 - k * x1;   l[n++] = {k, b};      }    sort(l, l + n, rule);  // 排序方便选择出不同的直线    int res = 1;    for (int i = 1; i  1e-8 || fabs(l[i].b - l[i - 1].b) > 1e-8)  // 由于浮点数存在误差,所以用一个极小的数判断两数是否相等     res++;    cout << res + 20 << endl;  // + 20 为斜率不存在时的直线    return 0;}

题目答案

40257

D题:货物摆放

问题描述

小蓝有一个超大的仓库,可以摆放很多货物。现在,小蓝有n 箱货物要摆放在仓库,每箱货物都是规则的正方体。小蓝规定了长、宽、高三个互相垂直的方向,每箱货物的边都必须严格平行于长、宽、高。小蓝希望所有的货物最终摆成一个大的立方体。即在长、宽、高的方向上分别堆L、W、H 的货物;满足n = L × W × H。给定n,请问有多少种堆放货物的方案满足要求。例如,当n = 4 时,有以下6 种方案:1×1×4、1×2×2、1×4×1、2×1×2、2×2×1、4×1×1。请问,当n = 2021041820210418 (注意有16 位数字)时,总共有多少种方案?提示:建议使用计算机编程解决问题。

思路分析

由 n = L x W x H 分析出L、W和H均为n的约数,我们只需要把n的所有约数求出存储,最后逐个遍历,当 i * j * k == n 时(i、j、k为n的约数)即为一个可行方案。

题目代码

#include #include using namespace std;typedef long long LL;int main() {    LL n = 2021041820210418;    LL l, w, h;    vector v;    /* 存储n的所有约数 */    for (LL i = 1; i * i < n; i++) if (n % i == 0) {     v.push_back(i);     if (n / i != i)  // 判断不是完全平方数,否则会多加一个  v.push_back(n / i); }    int cnt = 0;    for (int i = 0; i < v.size(); i++) for (int j = 0; j < v.size(); j++)     for (int k = 0; k < v.size(); k++)  if (v[i] * v[j] * v[k] == n)      cnt++;    cout << cnt;    return 0;}

题目答案

2430

E题:路径

小蓝学习了最短路径之后特别高兴,他定义了一个特别的图,希望找到图中的最短路径。小蓝的图由2021 个结点组成,依次编号1 至2021。对于两个不同的结点a, b,如果a 和b 的差的绝对值大于21,则两个结点之间没有边相连;如果a 和b 的差的绝对值小于等于21,则两个点之间有一条长度为a 和b 的最小公倍数的无向边相连。例如:结点1 和结点23 之间没有边相连;结点3 和结点24 之间有一条无向边,长度为24;结点15 和结点25 之间有一条无向边,长度为75。请计算,结点1 和结点2021 之间的最短路径长度是多少。提示:建议使用计算机编程解决问题

问题描述

思路分析

这是一个简单的最短路问题,本题需要解决两个问题:求出可选择路径与编写最短路算法。本题作者选用Dijkstra算法。

其实也可以使用floyd算法,虽然时间复杂度为O(N^3),会算半分钟。但这是填空题。使用臭氧优化加快代码运行速度,最后也就是十多秒就可以算出

题目代码(Dijkstra)

#include #include using namespace std;const int N = 2030;int n = 2021, m;int g[N][N];  // 邻接矩阵,存储图int dist[N];  // 存储每个点到起点的最短距离bool st[N];   // 判断每个点到起点的最短距离是否确定int gcd(int a, int b) {  // 欧几里得算法    return b ? gcd(b, a % b) : a;}int dijstra() {    memset(dist, 0x3f, sizeof dist);  // 初始化距离为正无穷    dist[1] = 0;    for (int i = 0; i < n; i++) { int t = 0;  // t 表示当前正在确定距离的点 for (int j = 1; j  dist[j]))  t = j;  // 寻找当前结点到所有下一结点中距离最短的点 } st[t] = true;  // 1号点到t点的最短距离确定 for (int j = 1; j <= n; j++)     dist[j] = min(dist[j], dist[t] + g[t][j]);  // 用已经确定最短距离的点更新未确定的点    }    return dist[n];}int main() {    memset(g, 0x3f, sizeof g);  // 初始化距离为正无穷(极大值表示)    /* 寻找有效边 */    for (int i = 1; i < n; i++) for (int j = i + 1; j <= n; j++) {     if (j - i <= 21) {  g[i][j] = i * j / gcd(i, j);  // 添加路径距离     } }    int t = dijstra();    cout << t;    return 0;}

题目代码(Floyd)

#include #pragma GCC optimize(3)using namespace std;const int N = 2050,INF = 1e9;int d[N][N]; //邻接矩阵存图int n;void floyd()//floyd{for(int k = 1;k <= n;k++)for(int i =1 ;i<=n;i++)for(int j =1;j<=n;j++)d[i][j] = min(d[i][j],d[i][k]+d[k][j]);}int gcd(int a,int b)//最大公约数{return b?gcd(b,a%b):a;}int main(){n =2021;memset(d,0x3f,sizeof d); for(int i=1;i<=n;i++)for(int j=1;j<=n;j++){if(abs(j-i)<=21)d[i][j]=min(d[i][j],i*j/gcd(i,j));}floyd();printf("%d\n",d[1][2021]);return 0;}

题目答案

10266837