#CSP0001. [2023CSP-J1] 第一轮真题

[2023CSP-J1] 第一轮真题

一、单项选择题(共 15 题,每题 2 分,共计 30 分;每题有且仅有一个正确选项)

  1. C++C++ 中,下面哪个关键字用于声明一个变量,其值不能被修改?( )。

{{ select(1) }}

  • unsignedunsigned
  • constconst
  • staticstatic
  • mutablemutable
  1. 八进制数 12345670812345670_807654321807654321_8 的和为( )。

{{ select(2) }}

  • 22222221822222221_8
  • 21111111821111111_8
  • 22111111822111111_8
  • 22222211822222211_8
  1. 阅读下述代码,请问修改 datadatavaluevalue 成员以存储 3.143.14,正确的方式是( )。
union Data{
    int num;
    float value;
    char symbol;
};
union Data data;

{{ select(3) }}

  • data.value = 3.14;
  • value.data = 3.14;
  • data->value = 3.14;
  • value->data = 3.14;
  1. 假设有一个链表的节点定义如下:
struct Node { 
	int data;     
	Node* next;
};

现在有一个指向链表头部的指针:Node* head。如果想要在链表中插入一个新节点,其成员data的值为42,并使新节点成为链表的第一个节点,下面哪个操作是正确的?( )

{{ select(4) }}

  • Node* newNode = new Node; newNode->data = 42; newNode->next = head; head = newNode;
  • Node* newNode = new Node; head->data = 42; newNode->next = head; head = newNode;
  • Node* newNode = new Node; newNode->data = 42; head->next = newNode;
  • Node* newNode = new Node; newNode->data = 42; newNode->next = head;
  1. 根节点的高度为 11,一根拥有 20232023 个节点的三叉树高度至少为( )。

{{ select(5) }}

  • 66
  • 77
  • 88
  • 99
  1. 小明在某一天中依次有七个空闲时间段,他想要选出至少一个空闲时间段来练习唱歌,但他希望任意两个练习的时间段之间都有至少两个空闲的时间段让他休息,则小明一共有( )种选择时间段的方案。

{{ select(6) }}

  • 3131
  • 1818
  • 2121
  • 3333
  1. 以下关于高精度运算的说法错误的是( )。

{{ select(7) }}

  • 高精度计算主要是用来处理大整数或需要保留多位小数的运算。
  • 大整数除以小整数的处理的步骤可以是,将被除数和除数对齐,从左到右逐位尝试将除数乘以某个数,通过减法得到新的被除数,并累加商。
  • 高精度乘法的运算时间只与参与运算的两个整数中长度较长者的位数有关。
  • 高精度加法运算的关键在于逐位相加并处理进位。
  1. 后缀表达式 6 2 3 + - 3 8 2 / + * 2 ^ 3 + 对应的中缀表达式是( )

{{ select(8) }}

  • ((6 - (2 + 3)) * (3 + 8 / 2)) ^ 2 + 3
  • 6 - 2 + 3 * 3 + 8 / 2 ^ 2 + 3
  • (6 - (2 + 3)) * ((3 + 8 / 2) ^ 2) + 3
  • 6 - ((2 + 3) * (3 + 8 / 2)) ^ 2 + 3
  1. 1010102101010_21668166_8 的和为( )。

{{ select(9) }}

  • 10110000210110000_2
  • 2368236_8
  • 15810158_{10}
  • A016A0_{16}
  1. 假设有一组字符 a,b,c,d,e,f{a,b,c,d,e,f},对应的频率分别为 5%9%12%13%16%45%5\%,9\%,12\%,13\%,16\%,45\%。请问以下哪个选项是字符 a,b,c,d,e,fa,b,c,d,e,f 分别对应的一组哈夫曼编码?( )

{{ select(10) }}

  • 1111111010110011001111,1110,101,100,110,0
  • 101010011000011010001010,1001,1000,011,010,00
  • 0000010100111011000,001,010,011,10,11
  • 1010101111011100011010,1011,110,111,00,01
  1. 给定一棵二叉树,其前序遍历结果为:ABDECFGABDECFG,中序遍历结果为:DEBACFGDEBACFG。请问这棵树的正确后序遍历结果是什么?( )

{{ select(11) }}

  • EDBFGCAEDBFGCA
  • EDBGCFAEDBGCFA
  • DEBGFCADEBGFCA
  • DBEGFCADBEGFCA
  1. 考虑一个有向无环图,该图包括 44 条有向边:(1,2)(1,2)(1,3)(1,3)(2,4)(2,4),和 (3,4)(3,4)。以下哪个选项是这个有向无环图的一个有效的拓扑排序?( )

{{ select(12) }}

  • 42314,2,3,1
  • 12341,2,3,4
  • 12431,2,4,3
  • 21342,1,3,4
  1. 在计算机中,以下哪个选项描述的数据存储容量最小?( )

{{ select(13) }}

  • 字节(bytebyte
  • 比特(bitbit
  • 字(wordword
  • 千字节(kilobytekilobyte
  1. 一个班级有 1010 个男生和 1212 个女生。如果要选出一个 33 人的小组,并且小组中必须至少包含 11 个女生,那么有多少种可能的组合?( )

{{ select(14) }}

  • 14201420
  • 17701770
  • 15401540
  • 22002200
  1. 以下哪个不是操作系统?( )

{{ select(15) }}

  • LinuxLinux
  • WindowsWindows
  • AndroidAndroid
  • HTMLHTML

二、阅读程序(程序输入不超过数组或字符串定义的范围;判断题正确填√,错误填×;除特殊说明外,判断题 1.5 分,选择题 3 分,共计 40 分)

第一大题

01 #include<iostream>
02 #include<cmath>
03 using namespace std;
04 
05 double f(double a,double b,double c){
06     double s=(a+b+c)/2;
07     return sqrt(s*(s-a)*(s-b)*(s-c));
08 }
09
10 int main(){
11    cout.flags(ios::fixed);
12    cout.precision(4);
13    
14     int a,b,c;
15    cin>>a>>b>>c;
16    cout<<f(a,b,c)<<endl;
17    return 0;
18 }

假设输入的所有数都为不超过 10001000 的正整数,完成下面的判断题和单选题:

判断题

  1. (2分)当输入为 2 2 22\ 2\ 2 时,输出为 1.73211.7321( ) {{ select(16) }}
  • 正确
  • 错误
  1. (2分)将第 77 行中的 (sb)(sc)(s-b) * (s-c) 改为 (sc)(sb)(s-c) * (s-b) 不会影响程序运行的结果( ) {{ select(17) }}
  • 正确
  • 错误
  1. (2分)程序总是输出四位小数( ) {{ select(18) }}
  • 正确
  • 错误

单选题

  1. (3分)当输入为 3 4 53\ 4\ 5 时,输出为( ) {{ select(19) }}
  • 6.00006.0000
  • 12.000012.0000
  • 24.000024.0000
  • 30.000030.0000
  1. (3分)当输入为 5 12 135\ 12\ 13时,输出为( ) {{ select(20) }}
  • 24.000024.0000
  • 30.000030.0000
  • 60.000060.0000
  • 120.0000120.0000

第二大题)

01 #include<iostream>
02 #include<vector>
03 #include<algorithm>
04 using namespace std;
05 
06 int f(string x,string y){
07     int m=x.size();
08     int n=y.size();
09     vector<vector<int>>v(m+1,vector<int>(n+1,0));
10    for(int i=1;i<=m;i++){
11        for(int j=1;j<=n;j++){
12             if(x[i-1]==y[j-1]){
13                 v[i][j]=v[i-1][j-1]+1;
14             }else{
15                 v[i][j]=max(v[i-1][j],v[i][j-1]);
16             }
17         }
18     }
19     return v[m][n];
20 }
21
22 bool g(string x,string y){
23     if(x.size() != y.size()){
24         return false;
25     }
26     return f(x+x,y)==y.size();
27 }
28
29 int main(){
30     string x,y;
31     cin>>x>>y;
32     cout<<g(x,y)<<endl;
33     return 0;
34 }

判断题

  1. (1.5分)ff 函数的返回值小于等于 min(n,m)min(n,m)。( )

{{ select(21) }}

  • 正确
  • 错误
  1. (1.5分) ff 函数的返回值等于两个输入字符串的最长公共子串的长度。( ) {{ select(22) }}
  • 正确
  • 错误
  1. (1.5分)当输入两个完全相同的字符串时,gg 函数的返回值总是 truetrue( ) {{ select(23) }}
  • 正确
  • 错误

单选题

  1. (3分)将第19行中的 v[m][n]v[m][n] 替换为 v[n][m]v[n][m],那么该程序( ) {{ select(24) }}
  • 行为不变
  • 只会改变输出
  • 一定非正常退出
  • 可能非正常退出
  1. (3分)当输入为 ccspjpjcsccsp-j p-jcs 时,输出为:( )

{{ select(25) }}

  • 00
  • 11
  • TT
  • FF
  1. (3分)当输入为 csppsc spsccpcsppsc\ spsccp 时,输出为:( )

{{ select(26) }}

  • TT
  • FF
  • 00
  • 11

第三大题(15分)

01 #include <iostream>
02 #include <cmath>
03 using namespace std;
04 
05 int solve1(int n){
06     return n*n;
07 }
08
09 int solve2(int n){
10    int sum=0;
11    for(int i=1;i<=sqrt(n);i++){
12         if(n%i==0){
13             if(n/i==i){
14                 sum+=i*i;
15             }else{
16                 sum+=i*i+(n/i)*(n/i);
17             }
18         }
19     }
20     return sum;
21 }
22
23 int main(){
24     int n;
25     cin>>n;
26     cout<<solve2(solve1(n))<<" "<<solve1((solve2(n)))<<endl;
27     return 0;
28 }

假设输入的 nn 是绝对值不超过 10001000 的整数,完成下面的判断题和单选题。

判断题

  1. 如果输入的 nn 为正整数,solve2solve2 函数的作用是计算 nn 所有的因子的平方和( )

{{ select(27) }}

  • 正确
  • 错误
  1. (2分) 第 131413 \sim 14 行的作用是避免 nn 的平方根因子ii(或n/in/i)进入第 1616 行而被计算两次( )

{{ select(28) }}

  • 正确
  • 错误
  1. (2分)如果输入的 nn 为质数,solve2(n)solve2(n) 的返回值为 n2+1n^2+1( ) {{ select(29) }}
  • 正确
  • 错误

单选题

  1. (4分)如果输入的 nn 为质数 pp 的平方,那么 solve2(n)solve2(n) 的返回值为( )

{{ select(30) }}

  • p2+p+1p^2+p+1
  • n2+n+1n^2+n+1
  • n2+1n^2+1
  • p4+2p2+1p^4+2p^2+1
  1. (3分)当输入为正整数时,第一项减去第二项的差值一定( )

{{ select(31) }}

  • 大于 00
  • 大于等于 00 且不一定大于 00
  • 小于 00
  • 小于等于 00 且不一定小于 00
  1. (3分) 当输入为 55 时,输出为( )

{{ select(32) }}

  • 651.625651.625
  • 650.729650.729
  • 651.676651.676
  • 652.625652.625

三、完善程序(单选题,每小题 3 分,共计 30 分)

第一大题(15分)

(寻找被移除的元素)问题:原有长度为 n+1n+1 公差为 11 等升数列,将数列输到程序的数组时移除了一个元素,导致长度为 nn 的开序数组可能不再连续,除非被移除的是第一个或最后之个元素。需要在数组不连续时,找出被移除的元素。试补全程序。

01  #include <iostream>
02  #include <vector>
03
04  using namespace std;
05
06  int find missing(vector<int>& nums) {
07 		int left = 0, right = nums.size() - 1;
08 		while (left < right){
09   		int mid = left + (right - left) / 2;
10   		if (nums[mid] == mid +  ___ ① ___){
11   	    	___ ② ___;
12    		}else{
13      		___ ③ ___;
14    		}
15   	}
16  	return ___ ④ ___;
17  }
18
19 	int main() {
20  	int n;
21 		cin >> n;
22 		vector<int> nums(n);
23 		for (int i= 0; i< n; i++) cin >> nums[i];
24 		int missing_number = find_missing(nums);
25 		if (missing_number == ___ ⑤ ___) {
26     		cout << "Sequence is consecutive" << endl;
27 		}else{
28    		cout << "Missing number is " << missing_numbeer << endl;
29 		}
30 		return 0;
31 	}
  1. ①处应填( )

{{ select(33) }}

  • 11
  • nums[0]nums[0]
  • rightright
  • leftleft
  1. ②处应填( )

{{ select(34) }}

  • left=mid+1left=mid+1
  • right=mid1right=mid-1
  • right=midright=mid
  • left=midleft=mid
  1. ③处应填( )

{{ select(35) }}

  • left=mid+1left=mid+1
  • right=mid1right=mid-1
  • right=midright=mid
  • left=midleft=mid
  1. ④处应填( )

{{ select(36) }}

  • left+nums[0]left+nums[0]
  • right+nums[0]right+nums[0]
  • mid+nums[0]mid+nums[0]
  • right+1right+1
  1. ⑤处应填( )

{{ select(37) }}

  • nums[0]+nnums[0]+n
  • nums[0]+n1nums[0]+n-1
  • nums[0]+n+1nums[0]+n+1
  • nums[n1]nums[n-1]

第二大题(15分)

(编辑距离)给定两个字符串,每次操作可以选择删除(Delete)、插入(Insert)、替换(Replace),一个字符,求将第一个字符串转换为第二个字符串所需要的最少操作次数。

1 	#include <iostream>
2 	#include <string>
3 	#include <vector>
4 	using namespace std;
5 
6 	int min(int x, int y, int z){
7 		return min(min(x, y), z);
8 	}
9 
10 	int edit_dist_dp(string str1, string str2){
11 		int m = str1.length();
12		int n = str2.length();
13		vector<vector<int>> dp(m + 1,vector<int>(n + 1));
14
15		for(int i = 0; i <= m; i++){
16			for(int j = 0; j <= n; j++){
17				if(i == 0)
18					dp[i][j] = ___ ① ___;
19				else if(j == 0)
20					dp[i][j] = ___ ② ___;
21				else if(___ ③ ___)
22					dp[i][j] = ___ ④ ___;
23				else
24					dp[i][j] = 1 + min(dp[i][j - 1], dp[i - 1][j], ___ ⑤ ___); 
25			}
26		}
27		return dp[m][n];
28	}
29	
30	int main(){
31		string str1, str2;
32		cin >> str1 >> str2;
33		cout << "Mininum number of operation:"
34			 << edit_dist_dp(str1, str2) << endl;
35		return 0; 
36	}
  1. ①处应填( )

{{ select(38) }}

  • jj
  • ii
  • mm
  • nn
  1. ②处应填( )

{{ select(39) }}

  • jj
  • ii
  • mm
  • nn
  1. ③处应填( )

{{ select(40) }}

  • str1[i1]==str2[j1]str1[i-1]==str2[j-1]
  • str1[i]==str2[j]str1[i]==str2[j]
  • str1[i1]!=str2[j1]str1[i-1]!=str2[j-1]
  • str1[i]!=str2[j]str1[i]!=str2[j]
  1. ④处应填( )

{{ select(41) }}

  • dp[i1][j1]+1dp[i-1][j-1]+1
  • dp[i1][j1]dp[i-1][j-1]
  • dp[i1][j]dp[i-1][j]
  • dp[i][j1]dp[i][j-1]
  1. ⑤处应填( )

{{ select(42) }}

  • dp[i][j]+1dp[i][j] + 1
  • dp[i1][j1]+1dp[i-1][j-1]+1
  • dp[i1][j1]dp[i-1][j-1]
  • dp[i][j]dp[i][j]