GESP 客观题评测系统

2024-12-Level-8

2024-12-Level-8

试卷解析总览,可直接查看每题答案与解析。

单选题(每题 2 分)

1 题(单选题

小杨家响应国家“以旧换新”政策,将自家的汽油车置换为新能源汽车,正在准备自编车牌。自编车牌包括5位数字或英文字母,要求第5位必须是数字,前4位中可以有最多1位英文字母。英文字母必须是大写,而且不能是O或I(因为容易与数字0或1混淆)。请问自编车牌共有多少种可能性?()。

A.
100,000
B.
1,060,000
C.
1,360,000
D.
1,460,000

正确答案B

解析详情

【答案】B

【考点】分类计数

【解析】 第5位必须是数字,有 10 种。前4位分两类:全是数字时有 10^4=10000 种;恰有1位字母时,位置有 4 种,字母有 24 种,其余3位数字有 10^3 种。 所以总数为 10×(10000+4×24×1000)=1,060,000,对应 B。

【易错点】 字母不能取 O 和 I,所以是 24 个,不是 26 个。

2 题(单选题

新年到,四家人在一起聚会。其中两家有三口人,另外两家有两口人。现在要安排大家在一张十人圆桌坐下,要求一家人必须相邻就座。由于有“主座”的习俗,每个座位都被认为是不同的。请问共有多少种就座方案?()。

A.
8640
B.
6912
C.
144
D.
60

正确答案A

解析详情

【答案】A

【考点】圆排列与捆绑法

【解析】 把四家人看成4个块,块大小分别为 3、3、2、2。先算块在有主座圆桌上的排法:选一个分界起点有 10 种,4 个块的环形顺序有 3! 种,但每种安排会被 4 个块边界重复计数一次,所以块排法为 10×4!/4=60。 每个3口之家内部有 3! 种,两家共 3!×3!;每个2口之家内部有 2! 种,两家共 2!×2!。总数 60×3!×3!×2!×2!=8640,选 A。

【易错点】 “有主座”说明旋转后座位不同,不能直接套无标号圆排列的 (n-1)!。

3 题(单选题

下面关于C++类继承的说法,错误的是()。

A.
一个类可以继承多个类。
B.
一个类可以被多个类继承。
C.
一个类可以继承另一个类的子类。
D.
抽象类必须被至少一个类继承,否则会编译错误。

正确答案D

解析详情

【答案】D

【考点】C++ 继承与抽象类

【解析】 A、B、C 都是允许的:C++ 支持多重继承,一个基类也能被多个派生类继承,继承关系还可以继续往下延伸。 D 错在“必须被继承”。抽象类即使没有任何派生类,也不会编译错误;只有试图直接实例化抽象类时才会报错。

【易错点】 抽象类的限制是“不能实例化”,不是“必须马上被继承”。

4 题(单选题

使用邻接表表达一个简单有向图,图中包含 v 个顶点、e 条边,则该出边表中边节点的个数为()。

A.
v×(v1)v \times (v - 1)
B.
v×vv \times v
C.
2×e2 \times e
D.
e

正确答案D

解析详情

【答案】D

【考点】有向图邻接表

【解析】 出边邻接表中,每条有向边只会出现在其起点的链表里一次。 因此图里有 e 条边,就对应 e 个边结点,选 D。2e 是无向图邻接表常见的边结点数。

【易错点】 有向图与无向图的邻接表存储次数不同,不要把 2e 套过来。

5 题(单选题

以下将二维数组作为参数的函数声明,哪个是符合语法的?()。

A.
void Bubble(int a[10][], int m);
B.
void Bubble(int a[][], int n, int m);
C.
void Bubble(int (*a)[20], int n);
D.
void Bubble(int * a[20], int n);

正确答案C

解析详情

【答案】C

【考点】二维数组参数

【解析】 把二维数组作为参数时,除第一维外,其余各维大小必须已知。`int (*a)[20]` 表示“指向含 20 个 int 的数组的指针”,能正确接收二维数组的一行地址。 A 少了第二维,B 两维都未定且语法损坏,D 是“含 20 个 int* 的数组”,不是二维数组,所以正确项是 C。

【易错点】 `int (*a)[20]` 和 `int *a[20]` 含义完全不同,括号位置不能看错。

6 题(单选题

已知两个点 A、B 在平面直角坐标系下的坐标分别为(xa,ya)(xa, ya)(xb,yb)(xb, yb),并分别定义变量 double xa,ya,xb,yb;存储坐标。假设直线 AB 的斜率存在,下列哪个表达式可以用来表达它?()。

A.
(xa - xb) / (ya - yb)
B.
(xa - xb) / (yb - ya)
C.
(ya - yb) / (xa - xb)
D.
(ya - yb) / (xb - xa)

正确答案C

解析详情

【答案】C

【考点】直线斜率公式

【解析】 直线 AB 的斜率是“纵坐标变化量 ÷ 横坐标变化量”,所以 k=(ya-yb)/(xa-xb)。 A 是倒数,B 分母符号错,D 等于斜率的相反数,因此应选 C。

【易错点】 斜率一定是 y 的差除以 x 的差,别把分子分母写反。

7 题(单选题

二项式(x+y)6(x+y)^6的展开式中x3y3x^3y^3项的系数是()。

A.
6
B.
15
C.
20
D.
120

正确答案C

解析详情

【答案】C

【考点】二项式定理

【解析】 在 (x+y)^6 中要得到 x^3y^3,等价于从 6 个因子里选 3 个提供 x,其余 3 个提供 y。 系数就是组合数 C(6,3)=20,所以选 C。

【易错点】 这里求的是组合数,不要再额外乘排列数。

8 题(单选题

以下关于动态规划的说法中,错误的是()。

A.
动态规划方法有递推和递归两种实现形式。
B.
递归实现动态规划方法的时间复杂度总是不低于递推实现。
C.
动态规划方法将原问题分解为一个或多个相似的子问题。
D.
动态规划方法通常能够列出递推公式。

正确答案B

解析详情

【答案】B

【考点】动态规划实现方式

【解析】 B 说“递归实现动态规划的时间复杂度总是不低于递推实现”太绝对。带记忆化的递归和递推版 DP 在很多问题里访问的状态数相同,时间复杂度可以一样。 A、C、D 都是常见正确描述,所以错误项是 B。

【易错点】 递归写法可能常数更大,但复杂度并不一定严格更差。

9 题(单选题

在下面的程序中,使用整数表示一种组合。整数二进制表示的某一位为1,表示该位对应的数被选中,反之为0表示未选中。例如,从0 - 5这6个数中选出3个,则0b111000代表选中3,4,5三个数,0b011001代表选中0,3,4三个数。zuhe_next函数按组合对应的整数由大到小的顺序,求出组合c的下一个组合。横线处可以填入的是()。

int intlow2(int c) {
    return ___; // 在此处填入选项
}
int zuhe_next_incur(int c, int n, int l) {
    if (n == 1) return c;
    if ((c & (1 << 1)) == 0) {
        int d = intlow2(c);
        c = (c & ~d);
        c = (c | (d >> 1));
    } else {
        c = (c & ~(1 << 1));
        c = zuhe_next_incur(c, n - 1, l + 1);
        int d = intlow2(c);
        c = (c | (d >> 1));
    }
    return c;
}

// 从n个数中选m个,当前组合为c
int zuhe_next(int c, int n, int m) {
    return zuhe_next_incur(c, n, 0);
}
A.
((c - 1)^c)
B.
((c - 1)^c) + 1)
C.
((c - 1)^c) >> 1)
D.
((((c - 1)^c) + 1) >> 1)

正确答案D

解析详情

【答案】D

【考点】位运算与 lowbit

【解析】 (c-1)^c 会得到从最低位 1 开始向右的连续全 1 掩码。若要只保留原来最低的那一位 1,就先加 1 形成更高位的单独 1,再右移 1 位。 因此 `intlow2(c)` 应写成 `((((c - 1)^c) + 1) >> 1)`,选 D。

【易错点】 直接用 `(c-1)^c` 得到的是一串 1,不是单独的最低位 1。

10 题(单选题

下面程序的输出为()。

#include <iostream>
using namespace std;
int main() {
    int N = 15, cnt = 0;
    for (int x = 0; x + x + x <= N; x++)
        for (int y = x; x + y + y <= N; y++)
            for (int z = y; x + y + z <= N; z++)
                cnt++;
        cout << cnt << endl;
        return 0;
}
A.
174
B.
447
C.
816
D.
4096

正确答案A

解析详情

【答案】A

【考点】多重循环计数

【解析】 程序统计的是满足 0≤x≤y≤z 且 x+y+z≤15 的三元组个数。按循环条件完整枚举后,总计数为 174。 `cout` 和 `return` 不在三重循环内部,所以程序只输出一次最终的 `cnt`,对应 A。

【易错点】 容易误把输出语句当成在外层循环里,从而把中间结果也算进去。

11 题(单选题

下面最长公共子序列程序中,横线处应该填入的是()。

#define MAX(A, B) ((A) > (B)) ? (A) : (B))
#define MIN(A, B) ((A) < (B)) ? (A) : (B))
int dp[MAX_L + 1][MAX_L + 1];
int LCS(char str1[], char str2[]) {
    int len1 = strlen(str1);
    int len2 = strlen(str2);
    for (int i = 0; i < len1; i++)
        for (int j = 0; j < len2; j++)
            if (str1[i] == str2[j])
                dp[i + 1][j + 1] = dp[i][j] + 1;
        else
            // 在此处填入选项
        return dp[len1][len2];
    }
}
A.
dp[i + 1][j + 1] = dp[i][j + 1] + dp[i + 1][j]
B.
dp[i + 1][j + 1] = MIN(dp[i][j + 1], dp[i + 1][j])
C.
dp[i + 1][j + 1] = MAX(dp[i][j + 1], dp[i + 1][j])
D.
dp[i + 1][j + 1] = MAX(dp[i][j + 1], dp[i + 1][j]) + 1

正确答案C

解析详情

【答案】C

【考点】最长公共子序列 DP

【解析】 当 `str1[i] != str2[j]` 时,LCS 只能来自“去掉 str1[i]”或“去掉 str2[j]”这两个子问题,因此转移应取两者较大值。 所以应填 `dp[i + 1][j + 1] = MAX(dp[i][j + 1], dp[i + 1][j])`,选 C。

【易错点】 `+1` 只在字符相等时出现,不匹配时不能加。

12 题(单选题

下列Dijkstra算法中,横线处应该填入的是()。

typedef struct Edge {
    int in, out; // 从下标in顶点到下标out顶点的边
    int len; // 边长度
    struct Edge * next;
} Edge;
// v: 顶点个数, graph: 出边邻接表, start: 起点下标, dis: 输出每个顶点的最短距离
void dijkstra(int v, Edge * graph[], int start, int * dis) {
    const int MAX_DIS = 0x7fffff;
    for (int i = 0; i < v; i++)
        dis[i] = MAX_DIS;
    dis[start] = 0;
    int * visited = new int[v];
    for (int i = 0; i < v; i++)
        visited[i] = 0;
    visited[start] = 1;
    for (int t = 0; ; t++) {
        int min = MAX_DIS, minv = -1;
        for (int i = 0; i < v; i++) {
            if (visited[i] == 0 && min > dis[i]) {
                min = dis[i];
                minv = i;
            }
        }
        if (minv < 0)
            break;
        visited[minv] = 1;
        for (Edge * e = graph[minv]; e != NULL; e = e->next) {
            ___ // 在此处填入选项
        }
    }
    delete[] visited;
}
A.
1 if (dis[e->out] > e->len) 2 dis[e->out] = e->len;
B.
1 if (dis[e->out] > min + e->len) 2 dis[e->out] = min + e->len;
C.
1 if (dis[e->in] > e->len) 2 dis[e->in] = e->len;
D.
1 if (dis[e->in] > min + e->len) 2 dis[e->in] = min + e->len;

正确答案B

解析详情

【答案】B

【考点】Dijkstra 松弛操作

【解析】 当前已确定的顶点是 `minv`,其最短距离为 `min`。沿出边 `e` 松弛时,应检查经过 `minv` 到 `e->out` 是否更短,即 `dis[e->out] > min + e->len`。 因此更新语句应写成 B,目标点也是 `e->out` 而不是 `e->in`。

【易错点】 Dijkstra 更新的是“起点到邻点的累计距离”,不是只看当前边长。

13 题(单选题

假设图 graph 中顶点数 v、边数 e,上题程序的时间复杂度为()。

A.
O(e)O(e)
B.
O(v2)O(v^2)
C.
O(vlogv+e)O(v \log v + e)
D.
O((v+e)logv)O((v + e) \log v)

正确答案B

解析详情

【答案】B

【考点】Dijkstra 复杂度

【解析】 这份代码每轮都线性扫描全部 v 个顶点来找当前最小未访问点,这一步重复 v 次,代价是 O(v^2)。 遍历所有邻接表总共再花 O(e),合起来是 O(v^2+e),通常记作 O(v^2),选 B。

【易错点】 只有使用堆优化时,Dijkstra 才常写成 O((v+e)log v)。

14 题(单选题

下面的快速排序程序中,两处横线处分别应填入的是()。

void quick_sort(int a[], int n) {
    if (n <= 1)
        return;
    int pivot = 0, l = 0, r = n - 1;
    while (____) { // 在此处填入选项
        while (r > pivot && a[r] >= a[pivot])
            r--;
        if (r > pivot) {
            int temp = a[pivot];
            a[pivot] = a[r];
            a[r] = temp;
            pivot = r;
        }
        while (l < pivot && a[l] <= a[pivot])
            l++;
        if (l < pivot) {
            int temp = a[pivot];
            a[pivot] = a[l];
            a[l] = temp;
            pivot = l;
        }
    }
    quick_sort(a, pivot);
    quick_sort(____); // 在此处填入选项
}
A.
1 | l < r 2 | a + pivot + 1, n - pivot - 1
B.
1 | l < r 2 | a + pivot + 1, n - pivot
C.
1 | l <= r 2 | a + pivot + 1, n - pivot - 1
D.
1 | l <= r 2 | a + pivot + 1, n - pivot

正确答案A

解析详情

【答案】A

【考点】快速排序分区

【解析】 分区完成的停止条件是 `l < r` 不再成立,也就是左右指针相遇时结束。 枢轴最终落在 `pivot` 位置,右半段应从 `a + pivot + 1` 开始,长度为 `n - pivot - 1`,所以两空都对应 A。

【易错点】 右半部分长度要减去枢轴本身占的 1 个位置。

15 题(单选题

上题程序的时间复杂度为()。

A.
O(n)O(n)
B.
O(n2)O(n^{2})
C.
O(2n)O(2^{n})
D.
O(nlogn)O(n \log n)

正确答案D

解析详情

【答案】D

【考点】快速排序平均复杂度

【解析】 快速排序平均情况下每次划分比较均衡,递归深度约为 O(log n),每层处理总代价为 O(n)。 因此平均时间复杂度是 O(n log n),选 D。

【易错点】 快速排序最坏是 O(n^2),但题目按常规默认考查平均复杂度。

判断题(每题 2 分)

1 题(判断题

表达式 '3' + '5' 的结果为 '8',类型为 char。

正确答案错误

解析详情

【答案】错误

【考点】字符运算与类型提升

【解析】 `'3'` 和 `'5'` 参与运算时会先提升为 int,其编码值分别是 51 和 53,相加得到 104。结果既不是字符 `'8'`,类型也不是 char。

【易错点】 字符加法算的是编码值,不是把字符当十进制数字直接相加。

2 题(判断题

在C++语言中,可以在函数内定义结构体,但该结构体类型只能在该函数内使用。

正确答案正确

解析详情

【答案】正确

【考点】结构体作用域

【解析】 在函数体内部定义的结构体类型,其名字只在该函数的局部作用域内可见。函数外部无法直接使用这个类型名。

【易错点】 类型定义也受作用域约束,不只是普通变量才受作用域影响。

3 题(判断题

对n个元素的数组进行排序,快速排序和归并排序的平均时间复杂度都为O(nlogn)O(n\log n)。但快速排序存在退化情况,使得时间复杂度升高至O(n2)O(n^2);归并排序需要额外的空间开销。

正确答案正确

解析详情

【答案】正确

【考点】排序复杂度比较

【解析】 快速排序和归并排序的平均时间复杂度都可达 O(n log n)。同时,快速排序确实可能退化到 O(n^2),而归并排序通常需要额外辅助数组。 所以题目整体表述正确。

【易错点】 不要把快速排序的平均复杂度和最坏复杂度混为一谈。

4 题(判断题

二维数组的最后一维在内存中一定是连续的,但第一维在内存中可能不连续。

正确答案错误

解析详情

【答案】错误

【考点】二维数组内存布局

【解析】 普通二维数组在内存中是按行连续存放的,整块空间本身连续,因此最后一维连续,第一维对应的一整行一整行也连续排列。 所以“第一维可能不连续”这个说法错误。

【易错点】 要区分“二维数组”与“指针数组”,后者才可能出现行不连续。

5 题(判断题

使用 math.h 或 cmath 头文件中的函数,表达式log(1000)\log(1000)的结果类型为 double、值约为 3。

正确答案错误

解析详情

【答案】错误

【考点】对数函数

【解析】 `log(1000)` 在标准库里表示自然对数,值约为 6.9078,不是 3。返回类型是 double 这一点对,但数值判断错了。

【易错点】 只有以 10 为底时 `log10(1000)` 才等于 3,`log` 默认不是常用对数。

6 题(判断题

你有三种硬币,分别面值2元、5元和7元,每种硬币都有足够多。买一本书需要27元,则有8种硬币组合(组合与顺序无关,“1个2元+1个5元+1个2元”与“1个5元+2个2元”认为是同样的组合)可以正好付清,且不需要对方找钱。

正确答案正确

解析详情

【答案】正确

【考点】不定方程计数

【解析】 求非负整数解 `2a+5b+7c=27`。逐个枚举可得共有 8 组解,因此正好付清 27 元的组合数是 8。

【易错点】 题目按“组合”计数,不区分硬币出现顺序。

7 题(判断题

使用哈希函数f(x)=x%pf(x) = x \% p建立键值为 int 类型的哈希表,只要 p 取小于等于哈希表大小的素数,可保证不发生碰撞。

正确答案错误

解析详情

【答案】错误

【考点】哈希冲突

【解析】 哈希函数 `f(x)=x%p` 即使 p 取素数,也只能改善分布,不能保证无碰撞。只要两个整数相差 p 的倍数,它们的哈希值就相同。

【易错点】 素数模数常用于降低冲突概率,但不是“绝不冲突”的充分条件。

8 题(判断题

杨辉三角中的第n行、第m项,即为将二项式(a+b)n(a+b)^{n}展开后anmbma^{n-m}b^{m}项的系数。

正确答案错误

解析详情

【答案】错误

【考点】二项式系数与杨辉三角

【解析】 将 `(a+b)^n` 展开后,`a^(n-m)b^m` 的系数对应的是组合数 `C(n,m)`。若按常见编号方式把杨辉三角第一行记作第 0 行,则它位于第 n 行第 m+1 项,而不是题目说的“第 m 项”。

【易错点】 杨辉三角的“第几项”与幂指数里的 m 往往会差 1,取决于编号从 0 还是 1 开始。

9 题(判断题

判断图是否连通,可以通过广度优先搜索实现。

正确答案正确

解析详情

【答案】正确

【考点】广度优先搜索

【解析】 从任意一个顶点做 BFS,如果最终能访问到图中所有顶点,就说明图连通;否则不连通。 因此 BFS 可以用来判断图是否连通。

【易错点】 判断连通性时要看“是否访问到全部顶点”,不是只看搜索能否开始。

10 题(判断题

要求解一元二次方程x2+ax+b=0x^2 + ax + b = 0,需要先判断表达式a2b×40a^2 - b \times 4 \geq 0是否为真。

正确答案错误

解析详情

【答案】错误

【考点】判别式条件

【解析】 若只讨论实数解,`x^2+ax+b=0` 的判别式确实是 `a^2-4b`。但题目说“要求解方程就需要先判断该条件是否为真”过于绝对,因为并不是所有求解场景都只限于实数范围。 因此这个断言按题目表述应判为错误。

【易错点】 “判断是否有实数解”与“是否能求解方程”不是完全相同的表述。