Bootstrap

2024年6月GESPC++四级真题解析

一、单选题(每题2分,共30分)

题目123456789101112131415
答案

B

B

B

D

B

D

D

A

A

B

A

D

A

B

D

1.下列代码中,输出结果是( )
#include<bits/stdc++.h>

using namespace std;

int func(int x,int y){

int a=x,b=y;

int t;

t=a;

a=t;

b=t;

cout<<a<<" "<<b<<" ";

}

int main()

{

int c,d;

c=12;

d=24;

func(12,24);

cout<<c<<" "<<d<<endl;

}

A. 12 24 24 12

B. 24 12 12 24

C. 12 12 24 24

D. 24 24 12 12

【答案】B

【考纲知识点】函数的定义与调用

【解析】模拟一下func()函数的过程,可以发现它将变量x和y的值进行了交换,而传入的x、y的分别是12和24,交换之后便是24和12,因此前两个数输出24和12,后两个数字就是顺序输出的c、d的值,也就是12和24。

2.下面函数不能正常执行的是()

A. 
#include<iostream>

using namespace std;

int func()

{

//...

}

int main()

{

//...

}

B.
#include<iostream>

using namespace std;

int main()

{

func();

}

int func()

{

//...

}

C.
#include<iostream>

using namespace std;

int func()

{

//...

}

int main()

{

func();

}

D.
#include<iostream>

using namespace std;

int func();

int main()

{

func();

}

int func()

{

//...

}

【答案】B

【考纲知识点】函数的定义与调用

【解析】在调用函数之前,应该先定义函数。

3.下面程序输出的是()

#include<iostream>

using namespace std;

int func();

int main()

{

int i=2;

cout<<i<<endl;

for(int x=0;x<1;x++){

int i=10;

cout<<i<<endl;

}

i=i+1;

cout<<i<<endl;

{

i=i*i;

cout<<i<<endl;

}

}

A. 2 2 3 9

B. 2 10 3 9

C. 2 10 11 121

D. 2 10 3 100

【答案】B

【考纲知识点】形参与实参、作用域

【解析】注意for循环内部的变量i只作用于for循环内部,因此第二个输出的数应该是10;而第6行的变量i的作用域则不包含for循环内部的部分,因此第3、4个数的值取决于第6行变量i的初始值,模拟计算一下即可。

4.假设变量 a 的地址是0x6ffe14,下面程序的输出是( )。
#include<iostream>

using namespace std;

int main(){

int *p;

int a=10;

p=&a;

p++;

cout<<p<<endl;

}

A. 10

B. 0x6ffe14

C. 0x6ffe15

D. 0x6ffe18

【答案】D

【考纲知识点】C++指针类型的概念及基本应用

【解析】一个int变量占4字节,因此当指针p移动到下一个地址时,应当加上对应的字节数4。

5. 如果下列程序输出的地址是 0x6ffe00 ,则 cout<<a+1<<endl; 输出的是()
#include<iostream>

using namespace std;

int main(){

int a[2][3]={0};

cout<<a<<endl;

}
A. 0x6ffe04

B. 0x6ffe0C

C. 0x6ffe08

D. 0x6ffe00

【答案】B

【考纲知识点】C++指针类型的概念及基本应用

【解析】一个int变量占4字节,而a是一个2*3规模的二维数组,相当于是2个3长度的一维数组;a的地址是第0个数组的第0个元素的地址,而a+1的地址是第1个数组的第0个元素的地址;中间间隔了3个int变量,因此应当加上对应的总字节数12。

6.C++中,关于文件路径说法错误的是()

A. "GESP.txt":指定与当前工作目录中的程序文件相同目录中的 GESP.txt 文件

B. "../data/GESP.txt":指定与当前工作目录中的程序文件上一级目录下的 data 目录中的 GESP.txt 文件

C. "./data/GESP.txt":指定与当前工作目录中的程序文件同级目录下的 data 目录中的 GESP.txt 文件

D. "GESP.txt"是绝对路径
【答案】D

【考纲知识点】文件重定向

【解析】当“GESP.txt”文件前面没有其他前置路径的时候,其表示的意思就是与当前⼯作⽬录中的程序⽂件相同⽬录中的GESP.txt⽂件

7. 关于直接插入排序,下列说法错误的是()

A.插⼊排序的最好情况是数组已经有序,此时只需要进⾏n-1次⽐较 ,时间复杂度为O(n)  

B.最坏情况是数组逆序排序,此时需要进⾏n(n-1)/2次⽐较以及n-1次赋值操作(插⼊)

C.平均来说插⼊排序算法的复杂度为O(n2)

D.空间复杂度上 ,直接插⼊法是就地排序 ,空间复杂度为O(n)
【答案】D

【考纲知识点】排序算法

【解析】就地排序只需要O(1)的额外空间。

8.下列程序横线处 ,应该输⼊的是 ( )。
#include<iostream>

using namespace std;

int n,a[10001];

void swap(int &a,int &b)

{

int t=a;

a=b;

b=t;

}

int main(){

cin>>n;

for(int i=1;i<=n;i++)

cin>>a[i];

for(int i=n;i>1;i--)

for(int j=1;j<i;j++)

if(a[j]>a[j+1])

_____________;

for(int i=1;i<=n;i++)

cout<<a[i]<<" ";

cout<<endl;

return 0;

}

A. swap(a[j],a[j+1]);

B. swap(a[j-1],a[j]);

C. swap(a[j-1],a[j+1]);    

D. swap(&a[j-1],&a[j+1]);

【答案】A

【考纲知识点】排序算法

【解析】考察冒泡排序的过程,不断地把较大的值往后放,因此,如果a[j]>a[j+1]成立,说明a[j]是较大的值,应当通过交换操作来将a[j]换到后面去。

9.下⾯关于递推的说法不正确的是( )。

A.递推表现为⾃⼰调用自己

B.递推是从简单问题出发,⼀步步的向前发展,最终求得问题。是正向的

C.递推中,问题的n要求是在计算中确定,不要求计算前就知道n

D.斐波那契数列可以⽤递推实现求解

【答案】A

【考纲知识点】递推算法

【解析】自己调用自己是递归的表现,而不是递推,注意区分。

10.关于几种排序算法的说法 ,下⾯说法错误的是(  )。

A.选择排序不是⼀个稳定的排序算法

B.冒泡排序算法不是⼀种稳定的排序算法

C.插⼊排序是⼀种稳定的排序算法

D.如果排序前2个相等的数在序列中的前后位置顺序和排序后它们2个的前后位置顺序相同,则称为⼀种稳定的 排序算法

【答案】B

【考纲知识点】排序算法

【解析】在冒泡排序中,两个元素交换的条件是a[j]>a[j+1],如果两个元素的值相等,他们并不会发生交换,因此是一种稳定的排序算法。

11.数组{45,66,23, 1, 10,97,52,88,5,33}进⾏从⼩到⼤冒泡排序过程中 ,第⼀遍冒泡过后的序列是(  )。

A. {45,23, 1, 10,66,52,88,5,33,97}

B. {45,66, 1,23, 10,97,52,88,5,33}

C. {45,66,23, 1, 10,52,88,5,33,97}

D. {45,66,23, 1, 10,97,52,88,33,5}

【答案】A

【考纲知识点】排序算法

【解析】按照冒泡排序的过程模拟一遍即可,66会被一直交换到97的前面的位置,97会被一直交换到数组末尾的位置。

12.下⾯的排序算法程序中 ,横线处应该填⼊的是(  )。
int a[8]={ 2,3, 4, 5, 6,2,3,1};

for (int i=1;i<8;i++)

{

int key = a[i];

int j=i-1;

while(a[j]>key && j>=0)

{

________;

j -= 1;

}

a[j + 1]= key;

}

A. a[j]=a[j-1];

B. a[j]=a[j+1];

C. a[j+1]=a[j-1];

D. a[j+1]=a[j];

【答案】D

【考纲知识点】排序算法

【解析】考察插入排序的过程,不断地将a[i]与前面的数字比较,以便找到合适的位置插入进去;因此,在已排好序的部分中,所有比a[i]大的数字都要整体往右移动一个单位;若a[j]比a[i]大,那么它往右移动一个单位之后的位置是j+1,所以应该将a[j]赋值给a[j+1]。

13.下⾯的程序中 ,如果输⼊ 10 0 ,会输出( )。
#include<iostream>

using namespace std;

double Division(int a,int b)

{

if(b==0)

throw "Division by zero condition!";

else

return ((double)a/(double)b);

}

void func()

{

int len,time;

cin>>len>>time;

cout<<Division(len,time)<<endl;

}

int main(){

try{

func();

}

catch(const char* errmsg){

cout<<errmsg<<endl;

}

catch(const int errmsg){

cout<<errmsg<<endl;

}

return 0;

}

A. Division by zero condition !

B. 0

C. 10  

D. 100
【答案】A

【考纲知识点】函数的定义与调用、函数参数传递

【解析】输入len=10与time=0,分别传入Division()函数中的a、b,于是有a=10、b=0,条件b==0成立。

14.10条直线 ,最多可以把平⾯分为多少个区域(  )。
A. 55

B. 56

C. 54

D. 58

【答案】B

【考纲知识点】贪心

【解析】当第x条直线加入进来的时候,我们总有一种贪心的方法,让这条直线与前面的所有x-1条直线相交,每发生一次相交都会额外产生一个区域,因此会产生x-1个区域,再加上整个平面本身会被分割一次,所以会产生x个区域,于是10条直线能够分割出来的区域数量就是1+2+3+4+5+6+7+8+9+10=55,最后不要忘记整个平面本身最初也有一个区域,所以还要加1,答案是55+1=56。

15.下⾯程序中,如果语句cout<<p<<endl;输出的是0x6ffe00,则cout<<++p<<endl;输出的是 ()
int x[10][10][10]={{0}};

int *p;

p=&x[0][0][0];

cout<<p<<endl;

cout<<++p<<endl;

A. 0x6ffe0c

B. 0x6ffe09

C. 0x6ffe06

D. 0x6ffe04
【答案】D

【考纲知识点】C++指针类型的概念及基本应用

【解析】一个int变量占4字节,p指针指向了x数组的首地址,因此,在++p之后,p指针移动到了下一个元素的地址,需要加上对应的int变量的字节数。

二、判断题(每题2分,共20分)

题目12345678910
答案

×

×

1.int& a 和 &a 是一样的,都是取 a 的地址。

【答案】错误

【考纲知识点】C++指针类型的概念及基本运用

【解析】&a是取a的地址,而int& a 是引用变量,作用是获取某个变量的“引用”,即给某一个变量起一个别名。

2.以下代码不能够正确执行。
#include<iostream>

using namespace std;

int main(){

int a=20;

int &ra;

ra=&a;

cout<<ra<<endl;

}

【答案】正确

【考纲知识点】C++指针类型的概念及基本运用

【解析】同上一题,int& ra 是引用变量,需要在定义时设置一个初始值。

3.引用是一个指针常量。

【答案】正确

【考纲知识点】C++指针类型的概念及基本运用

【解析】指针常量指向的内存地址无法修改,而指向的内存地址对应的值可以修改。

4.下面程序两个输出结果是一样的。
#include<iostream>

using namespace std;

int main(){

int a[2][3]={0};

cout<<a<<endl;

cout<<&a[0][0]<<endl;

}

【答案】正确

【考纲知识点】C++指针类型的概念及基本运用

【解析】输出a即直接输出数组a的首地址,而a[0][0]是数组a的第0个元素,因此&a[0][0]相当于就是取了第0个元素的地址,即首地址。

5.函数不可以调用自己。

【答案】错误

【考纲知识点】函数的定义与使用

【解析】递归算法的实现就是“函数调用自己”。

6.函数参数传递过程中,如果传常量值、常量引用和常量指针都是不能被修改的,它们可以防止函数对实参的值或地址进行修改。

【答案】正确

【考纲知识点】C++指针类型的概念及基本运用

【解析】常量无法被修改,因此若想要实参的值不被修改,传入常量即可。7.下面代码输出的值等于0。
#include<iostream>

using namespace std;

int main(){

int *p=NULL;

cout<<p<<endl;

}

【答案】正确

【考纲知识点】C++指针类型的概念及基本运用

【解析】空指针指向0地址。

8. 在下面这个程序里, a[i][j] 和一个普通的整型变量一样使用。
#include<iostream>

using namespace std;

int main(){

int a[10][10]={0};

for(int i=0;i<10;i++){

for(int j=0;j<10;j++){

if(i==j){

a[i][j]=1;

}

}

}

}

【答案】正确

【考纲知识点】C++二维数组与多维数组的应用

【解析】整型二维数组中的每一个元素都是一个整型变量。

9. 一个一维数组,至少含有一个自然数N,是一个合法的数列。可以在一维数组末尾加入一个自然数M,M不能超过一维数组末尾元素的一半,形成一个新的合法的一维数组,如果N=6,那么可以有6个不同的合法数组。

【答案】正确

【考纲知识点】递推

【解析】合法的6个数组分别是{6}、{6,1}、{6,2}、{6,3}、{6,3,1}、{6,2,1}

10.插⼊排序算法中 ,平均时间复杂度是O(n2),最坏的情况逆序情况下 ,达到最⼤时间复杂度。

【答案】正确

【考纲知识点】排序算法

【解析】在最坏的逆序情况下,每一个元素都会被插入到已排好序部分的最前面。

三、编程题(每题25分,共50分)

1、黑白方块

【问题描述】

⼩杨有⼀个n⾏m列的⽹格图 ,其中每个格⼦要么是⽩⾊ ,要么是⿊⾊ 。

对于⽹格图中的⼀个⼦矩形 ,⼩杨认为它是平衡的当且仅当其中⿊⾊格⼦与⽩⾊格⼦数量相同。 ⼩杨想知道最⼤的平衡⼦矩形包含了多少个格⼦。

【输入描述】

第⼀⾏包含两个正整数n, m ,含义如题⾯所⽰ 。

之后n⾏ ,每⾏⼀个长度为 m的01串 ,代表⽹格图第 i⾏格⼦的颜⾊ ,如果为  ,则对应格⼦为⽩⾊ ,否则为⿊ ⾊ 。

【输出描述】

输出⼀个整数,代表最⼤的平衡⼦矩形包含格⼦的数量,如果不存在则输出0。

【样例输入 1】

4 5

00000

01111

00011

00011

【样例输出 1】

16

【题目大意】

【考纲知识点】
C++二维数组和多维数组的基本应用

【解题思路】

首先解决 “如何判断一个矩阵 a是一个平衡矩阵”这个问题:枚举矩阵中的每一个元素,用两个计数器a[0]、a[1]分别记录0和1的数量,如果最终a[0]==a[1],那么就是平衡矩阵,复杂度是O(n^2)。

然后解决“如何枚举一个矩阵中的所有子矩阵”这个问题:对于任意一个子矩阵,它都有一个起始行、一个终止行、一个起始列和一个终止列,我们可以用4个for循环来分别枚举它们,这样就应该把每个子矩阵确定下来,复杂度是O(n^4)。

如此一来,我们便可以用O(n^4)的复杂度枚举每一个子矩阵,然后再花O(n^2)的复杂度判断是否是平衡矩阵,总体时间复杂度为O(n^6)。

【参考程序】
#include<bits/stdc++.h>

using namespace std;

const int N = 55;

int w[N][N];

int n,m;

bool check(int xa,int ya,int xb,int yb){

int a[2]={0,0};

for(int i = xa;i<=xb;i++){

for(int j=ya;j<=yb;j++){

a[w[i][j]]++;

}

}

return a[0]==a[1];

}

int main(){

cin>>n>>m;

for(int i=1;i<=n;i++){

string s;

cin>>s;

for(int j=1;j<=m;j++){

w[i][j]=s[j-1]-'0';

}

}

int ans = 0;

for(int i=1;i<=n;i++){

for(int j=1;j<=m;j++){

for(int ii=i;ii<=n;ii++){

for(int jj=j;jj<=m;jj++){

if(check(i,j,ii,jj)){

ans = max(ans,(ii-i+1)*(jj-j+1));

}

}

}

}

}

cout<<ans<<"\n";

}

2、宝箱

【问题描述】

⼩杨发现了n个宝箱 ,其中第 i个宝箱的价值是ai。

⼩杨可以选择⼀些宝箱放⼊背包并带走 ,但是⼩杨的背包⽐较特殊 ,假设⼩杨选择的宝箱中最⼤价值为x,最⼩价值为y,⼩杨需要保证x - y ≤ k ,否则⼩杨的背包会损坏。

⼩杨想知道背包不损坏的情况下, ⾃⼰能够带⾛宝箱的总价值最⼤是多少。

【输入描述】

第⼀⾏包含两个正整数n, k ,含义如题⾯所⽰ 。

第⼆⾏包含n个正整数a1, a2 , · · · , an,代表宝箱的价值。

【输出描述】
输出⼀个整数 ,代表带走宝箱的最⼤总价值。

【样例输入 1】

5 1

1 2 3 1 2

【样例输出 1】

7

【题目大意】

【考纲知识点】
排序算法

【解题思路】

比较容易想到的做法是用O(n)的复杂度枚举一个a[i]作为最大值x,然后用O(n)的复杂度枚举另一个值a[j]作为最小值y,最后用O(n)的复杂度枚举数组,计算一下值处于[x,y]的元素总和sum,算出来的总和与ans取最大值即可,但是这样的做法是O(n^3)的,没有办法拿到满分;

我们发现,之所以我们最后需要O(n)的复杂度枚举数组的元素计算总和,是因为我们不知道数值处在[x,y]范围内的元素总和是多少,所以需要枚举计算。

为了解决这个问题,我们可以先将数组从小到大排序,这样一来,数组中的每个子区间[j,i]的左右端点的值a[j]、a[i]分别就对应了我们枚举的最小值x,和最大值y,又因为数组是有序的,因此,数值处于[x,y]范围内的元素正好就是区间[j,i]内的所有元素,这样一来,我们就可以在向前枚举a[j](也就是最小值y)的同时,顺便计算区间[j,i]内的元素总和sum,时间复杂度降为了O(n^2)。

【参考程序】
#include<bits/stdc++.h>

using namespace std;

const int N = 1010;

int a[N];

int n,k;

int main(){

cin>>n>>k;

for(int i=1;i<=n;i++){

cin>>a[i];

}

sort(a+1,a+n+1);

int ans=0;

for(int i=1;i<=n;i++){

int sum=0;

for(int j=i;j>=1;j--){

if(a[i]-a[j]<=k){

sum+=a[j];

}else break;

}

ans=max(ans,sum);

}

cout<<ans<<"\n";

}

;