#939. CSP2025年提高组初赛真题
CSP2025年提高组初赛真题
2025 CCF 非专业级别软件能力认证第一轮
(CSP-S1) 提高级 C++语言试题
认证时间: 2025年9月20日 14:30~16:30
考生注意事项:
- 试题共有14页,答题纸共有1页,满分100分。请在答题纸上作答,写在试题纸上的一律无效。
- 不得使用任何电子设备(如计算器、手机,电子词典,电子手表等)或查阅任何书籍资料。
一、单项选择题(共15题,每题2分,共计30分;每题有且仅有一个正确选项)
- 有5个红色球和5个蓝色球,它们除了颜色这外完全相同。将这10个球排成一排,要求任意两个蓝色球都率能相邻,有多少种不同的排列方法? ( ) {{ select(1) }}
- 25
- 30
- 6
- 120
- 在KMP算法中,对于模式串 P="abacaba",其next数组(next[i]定义为模式串P[0... i]最长公共前后缀的长度,且数组下标从0开始)的值是什么?( ) {{ select(2) }}
- {0, 0, 1, 0, 1, 2, 3}
- {0, 0, 1, 0, 1, 2, 3}
- { 0 , 0} 1, 1, 2, 2, 3}
- {0, 0, 0, 0, 1, 2, 3}
- 对一个大小为 16(下标 0-15)的数组上构建满线段树。查询区间[3,11]时,最少需要访问多少个树结点(包括路径上的交结点和完全包含在查询区间内的结点)?( ) {{ select(3) }}
- 7
- 8
- 9
- 10
- 将字符串"cat", "car", "cart", "case", "dog", "do" 插入一个空的 Trie 树 (前缀树)中。构建完成的 Trie 树(包括根结点)共有多少个结点?( ) {{ select(4) }}
- 8
- 9
- 10
- 11
- 对于一个包含 n 个顶点和 m 条边的有向无环图 (DAG),其拓扑排序的结果有多少种可能? ( ) {{ select(5) }}
- 只有1种
- 最多n种
- 等于n-m种
- 以上都不对
- 在一个大小为 13 的哈希表中,使用闭散列法的线性探查来解决冲突。哈希函数为H(key)= key mod 13。依次插入关键字 18, 26, 35, 9, 68, 74。插入 74 后, 它最终被放置在哪个索引位置? ( ) {{ select(6) }}
- 5
- 7
- 9
- 11
- 一个包含 8 个顶点的完全图(顶点的编号为1到8),任意两点之间的边权重等于两顶点编号的差的绝对值。例如,顶点 3 和 7 之间的边权重为 |7− 3|=4。该图的最小生成树的总权重是多少? ( ) {{ select(7) }}
- 7
- 8
- 9
- 10
- 如果一棵二叉搜索树的后序遍历序列是 2,5,4,8,12,10,6,那么该树的前序遍历序列是什么? ( ) {{ select(8) }}
- 6, 4, 2, 5, 10, 8, 12
- 6, 4, 5, 2, 10, 12, 8
- 2, 4, 5, 6, 8, 10, 12
- 12, 8, 10, 5, 2, 4, 6
- 一个0-1背包问题,背包容量为 20。现有5个物品,其重量和价值分别为7,5,4,3,6和15,12,9,7,13。装入背包的物品能获得的最大总价值是多少?( ) {{ select(9) }}
- 43
- 41
- 45
- 44
- 在一棵以结点 1 为根的树中,结点 12 和结点 18 的最近公共祖先 (LCA)是结点4。那么下列哪个结点的 LCA 组合是不可能出现的? ( ) {{ select(10) }}
- LCA(12, 4) = 4
- LCA(18, 4) = 4
- LCA(12, 18, 4) = 4
- LCA(12, 1) = 4
- 递归关系式 描述了某个分治算法的时间复杂度。请问该算法的时间复杂度是多少?( ) {{ select(11) }}
- 在一个初始为空的最小堆(min-heap) 中, 依次插入元素 20, 12, 15, 8, 10, 5。然后连续执行两次“删除最小值”(delete-min)操作。请问此时堆顶元素是什么? ( ) {{ select(12) }}
- 10
- 12
- 15
- 20
- 1到1000之间,不能被2、3、5中任意一个数整除的整数有多少个? ( ) {{ select(13) }}
- 266
- 267
- 333
- 734
- 斐波那契数列的定义为 F(0)=0, F(1)=1, F(n)=F(n-1)+F(n-2)。 使用朴素递归方法计算F(n)的时间复杂度是指数级的。而使用动态规划(或迭代)方法的时间复杂度是线性的。造成这种巨大差异的根本原因是? ( ) {{ select(14) }}
- 递归函数调用栈开销过大
- 操作系统对递归深度有限制
- 朴素递归中存在大量的重叠子问题未被重复利用
- 动态规划使用了更少的数据存储空间
- 有 5 个独立的,不可抢占的任务A1,A2,A3,A4,A5 需要在一台机器上执行(从时间0开始执行),每个任务都有对应的处理时长和截止时刻,按顺序分别为3,4,2,5,1和5,10,3,15,11。如果某一个任务超时,相应的惩罚等于其处理时长。为了最小化总惩罚,应该优先执行哪个任务? ( ) {{ select(15) }}
- 处理时间最短的任务 A5
- 截止时间最早的任务 A3
- 处理时间最长的任务 A4
- 任意一个任务都可以
二、阅读程序(程序输入不超过数组或字符串定义的范围;判断题正确填v,错误填x;除特殊说明外,判断题1.5分,选择题3分,共计40分)
(1)
01 #include<algorithm>
02 #include<cstdio>
03 #include<cstring>
04 bool flag[27];
05 int n;
06 int p[27];
07 int ans = 0;
08 void dfs(int k ) {
09 if (k == n + 1) {
10 ++ans;
11 return ;
12 }
13 for (int i = 1; i <= n; ++i ) {
14 if (flag[i]) continue ;
15 if (k > 1 && i == p[k - 1] + 1) continue;
16 p[k] = i;
17 flag[i] = true;
18 dfs(k + 1);
19 flag[i] = false ;
20 }
21 return;
22 }
23 int main() {
24 scanf("%d", &n);
25 dfs(1);
26 printf("%d\n", ans) ;
27 return 0;
28 }
- 判断题
- (1分)当输入的n=3的时候,程序输出的答案为3。 ( ) {{ select(16) }}
- 正确
- 错误
- 在 dfs函数运行过程中, k的取值会满足。 ( ) {{ select(17) }}
- 正确
- 错误
- 删除第19行的 “flag[i]= false;”, 对答案不会产生影响。( ) {{ select(18) }}
- 正确
- 错误
- 单选题
- 当输入的n=4的时候,程序输出的答案为 ( )。 {{ select(19) }}
- 11
- 12
- 24
- 9
- 如果因为某些问题,导致程序运行第25行的 dfs函数之前,数组p的初值并不全为0,则对程序的影响是( )。 {{ select(20) }}
- 输出的答案比原答案要小
- 无法确定输出的答案
- 程序可能陷入死循环
- 没有影响
- 假如删去第14行的“if(flag[i]) continue;”, 输入3, 得到的输出答案是( ) 。 {{ select(21) }}
- 27
- 3
- 16
- 12
(2)
01 #include<algorithm>
02 #include<cstdio>
03 #include<cstring>
04 #define ll long long
05 int cnt_broken = 0;
06 int cnt_check = 0;
07 int n, k;
08 inline bool check(int h ) {
09 printf("now check:%d\n", h);
10 ++ cnt_check;
11 if (cnt_broken == 2) {
12 printf("You have no egg!\n");
13 return false ;
14 }
15 if (h >= k) {
16 ++ cnt_broken;
17 return true ;
18 } else {
19 return false ;
20 }
21}
22 inline bool assert_ans(int h) {
23 if (h == k) {
24 printf("You are Right using %d checks\n", cnt_check);
25 return true ;
26 } else {
27 printf("Wrong answer!\n");
28 return false ;
29 }
30}
31 inline void guess1 ( int n ) {
32 for (int i = 1; i <= n ; ++i) {
33 if ( check (i) ) {
34 assert_ans(i);
35 return;
36 }
37 }
38}
39 inline void guess2 (int n ) {
40 int w = 0;
41 for (w = 1; w * (w + 1) / 2 < n; ++w)
42 ;
43 for (int ti = w, nh = w;; --ti, nh += ti, nh = std:: min(nh, n)) {
44 if ( check (nh ) ) {
45 for (int j = nh - ti + 1; j < nh; ++j ) {
46 if ( check(j) ) {
47 assert_ans(j);
48 return ;
49 }
50 }
51 assert_ans(nh);
52 return;
53 }
54 }
55}
56 int main( ) {
57 scanf("%d%d", &n, &k);
58 int t ;
59 scanf("%d", &t);
60 if ( t == 1) {
61 guess1(n);
62 } else {
63 guess2(n);
64 }
65 return 0;
66}
(注意:下述的“猜测数”为调用 check 函数的次数(即 cnt_check 的值); “猜测正确”的含义为assert_ans 函数 return true(执行第25行所在分支) 的情况; 所有输入保证 1 ≤ k ≤ n。)
- 判断题
- 当输入为“6 5 1”时, 猜测次数为5; 当输入“6 5 2”时, 猜测次数为3。( ) {{ select(22) }}
- 正确
- 错误
- 不管输入的n和k 具体为多少,t=2时的猜测数总是小于等于 t=1时的猜测数。 ( ) {{ select(23) }}
- 正确
- 错误
- 不管t=1或t=2,程序都一定会猜到正确结果。 ( ) {{ select(24) }}
- 正确
- 错误
- 单选题
- 函数guess1在运行过程中, cnt broken的值最多为( ) 。 {{ select(25) }}
- 0
- 1
- 2
- n
- 函数guess2在运行过程中,最多使用的猜测次数的量级为( )。 {{ select(26) }}
- 当输入的n=100的时候,代码中t=1和t=2分别需要的猜测次数最多分别为( )。 {{ select(27) }}
- 100,14
- 100,13
- 99,14
- 99,13
(3)
01 #include<algorithm>
02 #include<cstdio>
03 #include<cstring>
04 #include<vector>
05 # define ll long long
06 int n, m;
07 std:: vector< int>k, p;
08 inline int mpow(int x, int k) {
09 int ans = 1;
10 for (; k; k = k >> 1, x = x * x) {
11 if (k & 1)
12 ans = ans * x;
13 }
14 return ans;
15}
16 std:: vector< int> ans1, ans2;
17 int cnt1, cnt2;
18 inline void dfs( std::vector<int> & ans, int & cnt, int l, int r, int v ) {
19 if ( l > r ) {
20 ++ cnt;
21 ans.push_back(v);
22 return;
23 }
24 for ( int i = 1; i <= m ; ++i) {
25 dfs(ans, cnt, l + 1, r, v + k[l] * mpow(i, p[l]));
26 }
27 return;
28}
29 std:: vector< int> cntans1;
30 int main() {
31 scanf("%d%d", &n, &m);
32 k.resize(n + 1);
33 p.resize(n + 1);
34 for ( int i = 1; i <= n; ++ i ) {
35 scanf("%d%d", &k[i], &p[i]);
36 }
37 dfs(ans1, cnt1, 1, n >> 1, 0);
38 dfs(ans2, cnt2, (n >> 1) + 1, n, 0);
39 std:: sort(ans1. begin(), ans1.end());
40 int newcnt1 = 1;
41 cntans1.push_back(1);
42 for ( int i = 1; i < cnt1; ++ i ) {
43 if (ans1[i] == ans1[newcnt1 - 1]) {
44 ++ cntans1[newcnt1 - 1];
45 } else {
46 ans1[newcnt1++] = ans1[i] ;
47 cntans1.push_back(1);
48 }
49 }
50 cnt1 = newcnt1 ;
51 std::sort(ans2.begin( ), ans2.end( ) ) ;
52 int las = 0;
53 ll ans = 0;
54 for ( int i = cnt2 - 1 ; i >= 0; --i) {
55 for ( ; las < cnt1 && ans1[las] + ans2[i] < 0; ++ las)
56 ;
57 if ( las < cnt1 && ans1[las] + ans2[i] == 0)
58 ans += cntans1 [las] ;
59 }
60 printf ("%lld\n", ans) ;
61 return 0 ;
62}
- 判断题
- 删除第51行的“std:: sort(ans2.begin(),ans2.end());” 后,代码输出的结果不会受到影响。 ( ) {{ select(28) }}
- 正确
- 错误
- 假设计算过程中不发生溢出,函数 mpow(x,k)的功能是求出的取值。( ) {{ select(29) }}
- 正确
- 错误
- 代码中第39行到第50行的目的是为了将ans1数组进行“去重”操作。 ( ) {{ select(30) }}
- 正确
- 错误
- 单选题
- 当输入为“3 15 1 2 -1 2 1 2”时, 输 出 结 果 为 ( ) 。 {{ select(31) }}
- 4
- 8
- 0
- 10
- 记程序结束前数组元素的最大值为P,则该代码的时间复杂度是 ( )。 {{ select(32) }}
- 本题所求出的是( )。 {{ select(33) }}
- 满足a,b,c∈[1,m]的整数方程 解的数量
- 满足a,b,c∈[1,m]的整数方程 解的数量
- 满足∈[0,m]的整数方程解的数量
- 满足∈[1,m]的整数方程 解的数量
三、完善程序(单选题,每小题3分,共计30分)
(1)特殊最短路
给定一个含N个点、M条边的带权无向图,边杈非负。起点为S,终点为T。对于一条S到T的路径,可以在整条路径中,至多选择一条边作为“免费边”:当第一次经过这条被选中的边时,费用视为0;如果之后再次经过该边,则仍按其原始权重计费。点和边均允许重复经过。求从S到T的最小总费用。
以下代码求解了上述问题。试补全程序。
01 #include<algorithm>
02 #include<iostream>
03 #include<queue>
04 #include<vector>
05 using namespace std;
06
07 const long long INF = 1e18;
08
09 struct Edge {
10 int to;
11 int weight;
12};
13
14 struct State {
15 long long dist;
16 int u;
17 int used_freebie;//0 for not used,1 for used
18 bool operator>(const State &other)const {
19 return dist > other.dist;
20 }
21};
22
23 int main() {
24 int n, m, s, t;
25 cin >> n >> m >> s >> t;
26
27 vector<vector<Edge>>adj(n + 1);
28 for (int i = 0; i < m; ++i) {
29 int u, v, w;
30 cin >> u >> v >> w;
31 adj[u].push_back({v, w});
32 adj[v].push_back({u, w});
33 }
34
35 vector<vector<long long>>d(n + 1, vector<long long>(2, INF));
36 priority_queue<State, vector<State>, greater<State >> pq;
37
38 d[s][0] = 0;
39 pq.push({0, s, ①});
40
41 while(!pq.empty()) {
42 State current = pq.top();
43 pq.pop();
44
45 long long dist = current.dist;
46 int u = current.u;
47 int used = current.used_freebie;
48
49 if (dist >②) {
50 continue;
51 }
52
53 for (const auto &edge : adj[u]) {
54 int v = edge.to;
55 int w = edge.weight;
56
57 if (d[u][used] + w < ③ ) {
58 ③ = d[u][used] + w;
59 pg.push({ ③, v, used});
60 }
61
62 if (used == 0) {
63 if ( ④ < d[v][1]) {
64 d[v][1] = ④ ;
65 pq.push({d[v][1], v, 1});
66 }
67 }
68 }
69 }
70
71 cout << ⑤ << endl;
72 return 0;
73}
- ①处应填( ) {{ select(34) }}
- 0
- 1
- -1
- false
- ②处应填( ) {{ select(35) }}
- d[u][!used]
- d[u][used]
- d[t][used]
- INF
- ③处应填( ) {{ select(36) }}
- d[v][1]
- d[v][used]
- d[u][used]
- d[v][0]
- ④处应填( ) {{ select(37) }}
- d[v][0]
- d[v][1]
- d[u][0]
- d[u][1]
- ⑤处应填( ) {{ select(38) }}
- d[t][1]
- d[t][0]
- min(d[t][0],d[t][1])
- d[t][0]+d[t][1]
(2) 最优测试
工厂有n条生产线(编号0~n-1),已知其中恰有一条生产线存在缺陷。
工厂打算通过客户反馈来间接测试生产线,从而找到存在缺陷的生产线。每一轮测试为,从若干生产线的产品取样混合成一个批次发给客户。若该批次中包含缺陷生产线的产品,客户将要求退货(结果记为1),否则正常收货(记为0)。受售后压力限制,在所有发货批次中,最多只能有k次退货(即结果为1的次数≤k)。工厂的目标是,设计最少的间接测试轮数w(发货总批次),保证根据客户收货或退货的反馈结果,唯一确定存在缺陷的生产线。
以下程序实现了工厂的目标,包含两部分:
-
i)确定w的最小值,并设计最优测试方案;
-
ii)根据测试结果推断存在缺陷的生产线。该程序确定w最小值的方法为:由于不同的生产线故障时,测试应当返回不同的结果,因此w轮测试的可能结果数不应少于生产线数量。
test_subset()函数为抽象测试接口,输入所有批次的方案并返回一个二进制编码;该编码表示为每批次的检测结果(即最低位是第1批次、最高位是第w批次);其实现不在此处给出。
试补全程序。
01 #include<algorithm>
02 #include<cstddef>
03 #include<iostream>
04 #include<vector>
05 using namespace std;
06
07 long long comb(int w, int i) {
08 if (i < 0 || i > w) {
09 return 0;
10 }
11 long long res = 1;
12 for (int t = 1; t <= i; ++t) {
13 res = res * (w - t + 1) / t;
14 }
15 return res;
16}
17
18 //计算长度为w、1的个数≤ k的码字总数
19 long long count_patterns(int w, int k) {
20 long long total = 0;
21 for (int t = 0; t <= min(w, k); ++t) {
22 total += comb(w, t);
23 }
24 return total;
25}
26
27 //抽象测试接口
28 int test_subset(const vector<vector<int>> &plan);
29
30 int solve(int n, int k) {
31 //===第1步:求最小w ===
32 int w = 1;
33 while (①) {
34 ++w;
35 }
36 cout << w << endl;
37
38 //===第2步:生成测试方案===
39 vector<vector<int>>code(n, vector<int>(w, 0));
40 int idx = 0;
41 for (int ones = 0; ones <= k && idx < n; ++ ones) {
42 vector<int>bits(w, 0);
43 fill(bits.begin(), bits.begin() + ones, 1);
44 do {
45 for (int b = 0; b < w; ++b) {
46 code[idx][b] = bits[b];
47 }
48 ++ idx;
49 if (idx >= n) {
50 break;
51 }
52 } while (std::②);
53 }
54
55 vector<vector<int>>plan(w);
56 for (int i = 0; i < w; ++i) {
57 for (int j = 0; j < n; ++j) {
58 if ( ③ ) {
59 plan[i].push_back(j);
60 }
61 }
62 }
63
64 //===第3步:调用测试接口===
65 int signature = test_subset(plan);
66
67 //===第4步:结果解码===
68 vector<int>sig_bits(w, 0);
69 for (int i = 0; i < w; ++i) {
70 if (④) {
71 sig_bits[i] = 1;
72 }
73 }
74
75 for (int j = 0; j < n; ++j) {
76 if (⑤) return j;
77 }
78}
79
80 int main() {
81 int n, k;
82 cin >> n >> k;
83 int ans = solve(n, k);
84 cout << ans << endl;
85 return 0;
86}
- ①处应填( ) {{ select(39) }}
- (1 << w) < n
- count_pattenns(w, k) < n
- count patterns(k, w) < n
- comb(w, k) < n
- ②处应填( ) {{ select(40) }}
- next_permutation(bits. begin(), bits.end())
- prev_permutation(bits.begin(), bits.end())
- next_permutation(bits. begin(), bits. begin()+ ones)
- prev_permutation(bits. begin(), bits.begin()+ ones)
- ③处应填( ) {{ select(41) }}
- ( j >> i ) & 1
- ( i >> j ) & 1
- code[i][j] == 1
- code[ j ] [ i ] == 1
- ④处应填( ) {{ select(42) }}
- (signature >> i) & 1
- (signature >> i) ^ 1
- signature | (1 << i)
- (signature >> i) | 1
- ⑤处应填( ) {{ select(43) }}
- is_permutation(code[j]. begin(), code[j]. end(), sig_bits. begin())
- code[j] == sig_bits
- plan [j] == sig_bits
- code[j][i] == sig_bits[i]