GESP 客观题评测系统

2025-09-Level-8

2025-09-Level-8

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

单选题(每题 2 分)

1 题(单选题

小杨想点一杯奶茶外卖,但还差5元起送。于是,小杨决定点一些小料。可选的小料包括:珍珠1元、椰果2元、奶冻3元、奶盖4元。每种小料最多点1份。请问共有多少种满足起送条件的点小料方案?()。

A.
16
B.
10
C.
9
D.
7

正确答案C

解析详情

【答案】C

【考点】枚举与分类计数

【解析】 4种小料每种选或不选,共有 2^4=16 种子集。金额不足5元的只有:不选、1、2、3、4、1+2、1+3,共7种。 所以满足起送条件的方案数是 16-7=9,对应 C。

【易错点】 把“每种最多1份”看成“可重复选”,会把方案数算多。

2 题(单选题

小杨和小刘是好朋友,她们在逛商场时发现新设置的大头贴自拍机,于是决定一起拍一组照片。一组照片包括4张,这4张照片没有顺序区分。拍每张照片时,可以选择有相框或无相框、两人可以分别选择有头饰或无头饰、还可以从2种位置(小杨在左,或小刘在左)中选出一种。她们不希望一组照片中出现完全相同的相框、头饰、位置的组合。请问一组照片共有多少种不同的方案?()。

A.
1820
B.
70
C.
24
D.
16

正确答案A

解析详情

【答案】A

【考点】组合计数

【解析】 单张照片的组合数为:相框 2 种,头饰对两人分别选择,共 2×2=4 种,站位 2 种,所以一共 2×4×2=16 种。 4张照片无顺序且不能重复,就是从16种里选4种,方案数为 C(16,4)=1820。

【易错点】 “4张没有顺序区分”说明应做组合,不是 16^4 或排列。

3 题(单选题

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

A.
派生类对象占用的内存总是不小于基类对象。
B.
派生类可以不实现基类的虚函数。
C.
如果一个类包含纯虚函数,则它不能包含成员变量。
D.
如果一个类包含纯虚函数,则不能用它定义对象。

正确答案C

解析详情

【答案】C

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

【解析】 含纯虚函数的类是抽象类,不能直接定义对象,但完全可以包含成员变量,所以 C 错。 A 对,因为派生类对象至少包含一个基类子对象;B 对,派生类若不重写纯虚函数或相关虚函数,自己仍可继续作为抽象类存在;D 也对。

【易错点】 把“抽象类不能实例化”误认为“抽象类内部什么成员都不能有”。

4 题(单选题

下列关于树和图的说法,错误的是()。

A.
每个连通图都存在生成树。
B.
每个存在生成树的有向图,都一定是强连通的。
C.
保留树的所有节点,并把树的每个节点指向其父节点,则可以将树转换为一个有向弱连通图。
D.
保留树的所有节点,并把树的每个节点指向其子节点,则可以将树转换为一个有向无环图。

正确答案B

解析详情

【答案】B

【考点】树、生成树与有向图性质

【解析】 A 正确,连通无向图一定存在生成树。B 错在把“存在生成树”误当成“强连通”;有向图只要存在从根到各点的生成树即可,并不要求任意两点互相可达。 C 中各点连向父节点后忽略方向仍连通,所以是弱连通图;D 中边都从父到子,不会形成环,因此是有向无环图。

【易错点】 强连通要求远强于“存在一棵有向生成树”,两者不能混淆。

5 题(单选题

一对夫妻生男生女的概率相同。这对夫妻希望儿女双全。请问这对夫妻生下三个孩子时,实现儿女双全的概率是多少?( )。

A.
14\frac{1}{4}
B.
12\frac{1}{2}
C.
34\frac{3}{4}
D.
78\frac{7}{8}

正确答案C

解析详情

【答案】C

【考点】古典概型

【解析】 3个孩子的性别共有 2^3=8 种等可能结果。儿女双全表示既有男孩也有女孩,可用反面计数:全男或全女共2种。 所以概率为 1-2/8=6/8=3/4,对应 C。

【易错点】 不要把“儿女双全”误算成“恰好一男一女”;这里是3个孩子。

6 题(单选题

二项式(x+y)6(x+y)^6的展开式中x2y4x^2y^4项的系数是()。

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

正确答案D

解析详情

【答案】D

【考点】二项式定理

【解析】 在 (x+y)^6 的展开式中,取 x^2y^4 说明要从6个因子里选2个提供 x,其余4个提供 y。 系数就是组合数 C(6,2)=15,所以选 D。

【易错点】 指数 2 和 4 只决定选取个数,不需要再乘 2!4!。

7 题(单选题

对一个包含 V 个顶点、E 条边的图,执行广度优先搜索,其最优时间复杂度是()。

A.
O(V)
B.
O(V + E)
C.
O(V²)
D.
O(E)

正确答案B

解析详情

【答案】B

【考点】广度优先搜索复杂度

【解析】 BFS 会把每个顶点至多入队、出队一次,因此顶点相关代价是 O(V)。 同时每条边也只会被检查常数次,所以总复杂度是 O(V+E),选 B。

【易错点】 不能只看队列操作写成 O(V),边扫描同样是复杂度的重要部分。

8 题(单选题

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

A.
动态规划能解决大部分多阶段决策问题。
B.
对特定的问题,贪心法不一定适用。
C.
当特定的问题适用贪心法时,通常比动态规划的时间复杂度更低。
D.
对很多问题,递推实现和递归实现动态规划方法的时间复杂度相当。

正确答案A

解析详情

【答案】A

【考点】贪心法与动态规划适用条件

【解析】 A 说“动态规划能解决大部分多阶段决策问题”表述过宽。动态规划必须有最优子结构和重叠子问题,不满足这些条件时并不能直接套用。 B 说明贪心并非总适用,正确;C 一般也对,适用贪心时常能比动态规划更省状态;D 中记忆化递归与递推版动态规划在很多题里时间复杂度相同。

【易错点】 动态规划很强,但不是“多阶段决策问题”的万能模板。

9 题(单选题

下面程序的输出为()。

#include <iostream>
using namespace std;
int main() {
    int N = 15, cnt = 0;
    for (int x = 1; 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.
45
B.
102
C.
174
D.
3375

正确答案B

解析详情

【答案】B

【考点】循环枚举与程序输出分析

【解析】 三重循环统计的是满足 1≤x≤y≤z 且 x+y+z≤15 的整数三元组个数。 按 x 分类:x=1 时有49组,x=2 时有30组,x=3 时有16组,x=4 时有6组,x=5 时有1组,总数 49+30+16+6+1=102。 因此程序输出 102,选 B。

【易错点】 cout 和 return 不在三重循环内部,程序只会输出一次总计数。

10 题(单选题

下面程序的时间复杂度为()。

int primes[MAXP], num = 0;
bool isPrime[MAXN] = {false};
void sieve() {
    for (int n = 2; n <= MAXN; n++) {
        if (!isPrime[n])
            primes[num++] = n;
        for (int i = 0; i < num && n * primes[i] <= MAXN; i++) {
            isPrime[n * primes[i]] = true;
            if (n % primes[i] == 0)
                break;
        }
    }
}
A.
O(nlogn)O(n \log n)
B.
O(nloglogn)O(n \log \log n)
C.
O(n)O(n)
D.
O(logn)O(\log n)

正确答案C

解析详情

【答案】C

【考点】线性筛时间复杂度

【解析】 这段代码是欧拉筛。关键性质是每个合数只会被它的最小质因子筛掉一次,所以内层循环的总执行次数对所有 n 累加后仍是线性的。 因此整体时间复杂度是 O(n),对应 C。

【易错点】 不要把双层循环机械地看成 O(n^2);要看内层是否真的对每个 n 都完整跑一遍。

11 题(单选题

下列Dijkstra算法,假设图 graph 中顶点数 v、边数 e,则程序的时间复杂度为()。

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) {
            if (dis[e->out] > e->len) {
                dis[e->out] = e->len;
            }
        }
    }
    delete[] visited;
}
A.
O(v^2)
B.
O(vlogv+e)
C.
O((v+e)logv)
D.
O(v+e)

正确答案A

解析详情

【答案】A

【考点】Dijkstra 实现复杂度分析

【解析】 这份程序每轮都用一个长度为 v 的线性扫描找当前未访问点中距离最小的顶点,这一步要做 v 次,所以代价是 O(v^2)。 遍历邻接表的总代价是 O(e),合起来是 O(v^2+e)。因为简单图中 e≤v^2,最终记为 O(v^2),选 A。

【易错点】 只有用堆优化时才常写成 O((v+e)logv);这段代码没有优先队列。

12 题(单选题

下面 count_triple 函数的时间复杂度为()。

int gcd(int m, int n) {
    if (m == 0) return n;
    return gcd(n % m, m);
}

int count_triple(int n) {
    int cnt = 0;
    for (int v = 1; v * v * 4 <= n; v++)
        for (int u = v + 1; u * (u + v) * 2 <= n; u += 2)
        {
            if (gcd(u, v) == 1) {
                int a = u * u - v * v;
                int b = u * v * 2;
                int c = u * u + v * v;
                cnt += n / (a + b + c);
            }
            return cnt;
        }
    }
}
A.
O(n2)O(n^2)
B.
O(n2logn)O(n^2 \log n)
C.
O(n)O(n)
D.
O(nlogn)O(n \log n)

正确答案D

解析详情

【答案】D

【考点】循环规模估计与欧几里得算法

【解析】 按题目意图分析,外层枚举 (u,v) 的总次数与 n 同阶;其中每次最重的操作是 gcd(u,v),欧几里得算法复杂度为 O(log n)。 因此总时间复杂度为 O(n log n),选 D。

【易错点】 这题不能只盯着两层循环个数,还要把 gcd 的对数复杂度一并乘进去。

13 题(单选题

下面 merge_sort 函数试图实现归并排序算法,横线处应该填入的是()。

#include <vector>
using namespace std;
void merge_sort(vector<int> & arr, int left, int right) {
    if (right - left <= 1)
        return;
    int mid = (left + right) / 2;
    merge_sort(____); // 在此处填入选项
    merge_sort(____); // 在此处填入选项
    vector<int> temp(right - left);
    int i = left, j = mid, k = 0;
    while (i < mid && j < right)
        if (arr[i] <= arr[j])
            temp[k++] = arr[i++];
        else
            temp[k++] = arr[j++];
    while (i < mid)
        temp[k++] = arr[i++];
    while (j < right)
        temp[k++] = arr[j++];
    for (i = left, k = 0; i < right; ++i, ++k)
        arr[i] = temp[k];
}
A.
arr, left, mid arr, mid, right
B.
arr, left, mid + 1 arr, mid + 1, right
C.
1 arr, left, mid 2 arr, mid + 1, right
D.
arr, left, mid + 1 arr, mid + 1, right + 1

正确答案A

解析详情

【答案】A

【考点】归并排序区间表示

【解析】 函数把排序区间写成左闭右开 [left, right)。因为归并时左半段用 [left, mid),右半段用 [mid, right),递归调用也必须保持同样的边界。 所以两处应分别填 arr, left, mid 和 arr, mid, right,即 A。

【易错点】 把右端点当成闭区间会导致区间重叠或漏元素。

14 题(单选题

下面Prim算法程序中,横线处应该填入的是()。

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
int prim(vector<vector<int>> & graph, int n) {
    vector<int> key(n, INT_MAX);
    vector<int> parent(n, -1);
    key[0] = 0;
    for (int i = 0; i < n; i++) {
        int u = min_element(key.begin(), key.end()) - key.begin();
        if (key[u] == INT_MAX)
            break;
        for (int v = 0; v < n; v++) {
            if (_____) { // 在此处填入选项
                key[v] = graph[u][v];
                parent[v] = u;
            }
        }
        int sum = 0;
        for (int i = 0; i < n; i++) {
            if (parent[i] != -1) {
                cout << "Edge: " << parent[i] << " - " << i << " Weight: " << key[i] << endl;
                sum += key[i];
            }
        }
        return sum;
    }
    int main() {
        int n, m;
        cin >> n >> m;
        vector<vector<int>> graph(n, vector<int>(n, 0));
        for (int i = 0; i < m; i++) {
            int u, v, w;
            cin >> u >> v >> w;
            graph[u][v] = w;
            graph[v][u] = w;
        }
        int result = prim(graph, n);
        cout << "Total weight of the minimum spanning tree: " << result << endl;
        return 0;
    }
}
A.
graph[u][v] >= 0 && key[v] > graph[u][v]
B.
graph[u][v] <= 0 && key[v] > graph[u][v]
C.
graph[u][v] == 0 && key[v] > graph[u][v]
D.
graph[u][v] != 0 && key[v] > graph[u][v]

正确答案D

解析详情

【答案】D

【考点】Prim 算法中的松弛条件

【解析】 邻接矩阵里 0 表示无边,因此候选边必须满足 graph[u][v] != 0。同时只有当当前边权更小,即 key[v] > graph[u][v] 时,才需要更新最小连接代价。 因此应填 graph[u][v] != 0 && key[v] > graph[u][v],选 D。

【易错点】 若写成 >=0,会把权值为0的“无边”也当成可选边,逻辑就错了。

15 题(单选题

下面的程序使用出边邻接表表达的带权无向图,则从顶点0到顶点3的最短距离为()。

#include <vector>
using namespace std;
class Edge {
public:
    int dest;
    int weight;
    Edge(int d, int w) : dest(d), weight(w) {}
};
class Graph {
    private:
        int num_vertex;
        vector<vector<Edge>> vve;
    public:
        Graph(int v) : num_vertex(v), vve(v) {}
        void addEdge(int s, int d, int w) {
            vve[s].emplace_back(d, w);
            vve[d].emplace_back(s, w)
        }
    };
    int main() {
        Graph g(4);
        g.addEdge(0, 1, 8);
        g.addEdge(0, 2, 5);
        g.addEdge(1, 2, 1);
        g.addEdge(1, 3, 3);
        g.addEdge(2, 3, 7);
        return 0;
    }
}
A.
12
B.
11
C.
10
D.
9

正确答案D

解析详情

【答案】D

【考点】最短路径计算

【解析】 图中从 0 到 3 的候选路径有:0→1→3 长度 8+3=11,0→2→3 长度 5+7=12,0→2→1→3 长度 5+1+3=9。 最短的是 9,所以选 D。

【易错点】 看到直接相连的边后容易停止比较,但最短路常需要经过中间顶点。

判断题(每题 2 分)

1 题(判断题

C++语言中,表达式 '9' ^ 3 的结果值为 '999'。

正确答案错误

解析详情

【答案】错误

【考点】字符常量与按位异或

【解析】 9 放在单引号里是字符常量,实际参与运算的是它的编码值 57。表达式 57 ^ 3 做的是按位异或,结果是数值 58,不会得到 999。

【易错点】 单引号表示单个字符,不是字符串。

2 题(判断题

下列C++语言代码,能够安全地输出 arr[5] 的值。

1 int n = 5; 2 int arr[n] = {1, 2, 3}; 3 std::cout << arr[5];

正确答案错误

解析详情

【答案】错误

【考点】数组越界

【解析】 arr 的有效下标只有 0 到 4,而 arr[5] 已经越界。越界访问属于未定义行为,因此这段代码不能安全输出 arr[5] 的值。

【易错点】 数组长度是5时,最后一个合法下标是4,不是5。

3 题(判断题

对n个元素的数组进行排序,最差情况的时间复杂度为O(n2)O(n^{2})

正确答案错误

解析详情

【答案】错误

【考点】排序算法复杂度

【解析】 题目没有限定具体排序算法,不能笼统说“排序的最差复杂度就是 O(n^2)”。例如归并排序、堆排序的最坏时间复杂度都是 O(n log n)。

【易错点】 把冒泡、选择、插入这类 O(n^2) 排序当成了所有排序算法的统一结论。

4 题(判断题

有4个红球、3个蓝球和2个绿球排成一排(相同色球视为完全相同),则不同的排列方案数为1260种。

正确答案正确

解析详情

【答案】正确

【考点】多重集合排列

【解析】 9个球中有4个红球相同、3个蓝球相同、2个绿球相同,不同排列数为 9!/(4!3!2!)=1260。 所以题目说法正确。

【易错点】 有重复元素时不能直接算 9!,必须除以相同颜色内部的重复排列。

5 题(判断题

使用 math.h 或 cmath 头文件中的函数,对于 int 类型的变量 x,表达式 fabs(x) 和 sqrt(x * x) 的结果总是近似相等的。

正确答案错误

解析详情

【答案】错误

【考点】数学函数与整型溢出

【解析】 在不溢出的情况下,fabs(x) 与 sqrt(x*x) 的数值确实相同;但题目说“总是”相等就错了。因为 x*x 先按 int 计算,x 较大时可能发生整型溢出,结果就不再可靠。

【易错点】 不要忽略表达式里先发生的整型乘法;问题往往出在 x*x 这一步。

6 题(判断题

运算符重载是C++语言静态多态的一种典型体现,而使用C语言则无法实现运算符重载。

正确答案正确

解析详情

【答案】正确

【考点】静态多态与运算符重载

【解析】 运算符重载发生在编译期,属于静态多态的典型形式。C 语言本身不支持用户自定义类型的运算符重载,因此题目说法正确。

【易错点】 静态多态不只有函数重载,运算符重载也属于这一类。

7 题(判断题

存在一个简单无向图满足:顶点数为6,边数为8,6个顶点的度数分别为3、3、3、3、2、2。

正确答案正确

解析详情

【答案】正确

【考点】图的度数序列

【解析】 度数和为 3+3+3+3+2+2=16,正好等于 2×8,与8条边相符。再用 Havel-Hakimi 检查序列 3,3,3,3,2,2,可继续化简到全0,因此这样的简单无向图确实存在。

【易错点】 度数和满足偶数只是必要条件,不是充分条件;还要看度数序列是否可图化。

8 题(判断题

已知两个 double 类型的变量 r 和 theta 分别表示一个扇形的圆半径及圆心角(弧度),则扇形的周长可以通过表达式(2 + theta) * r 求得。

正确答案正确

解析详情

【答案】正确

【考点】扇形周长公式

【解析】 扇形周长 = 两条半径 + 弧长 = 2r + rθ。提取公因子 r 后就是 (2 + θ)r,所以题目表达式正确。

【易错点】 弧度制下弧长公式是 rθ;若把 θ 当角度数,这个式子就不能直接用。

9 题(判断题

Dijkstra算法的时间复杂度为O(V2)O(V^{2}),其中 V 为图中顶点的数量。

正确答案错误

解析详情

【答案】错误

【考点】Dijkstra 算法复杂度

【解析】 Dijkstra 的时间复杂度取决于实现方式。用邻接矩阵加顺序找最小值时常是 O(V^2),但用堆优化和邻接表时常写成 O((V+E)logV),所以不能一概断言为 O(V^2)。

【易错点】 算法名相同,不代表不同数据结构实现的复杂度也完全相同。

10 题(判断题

从32名学生中选出2人分别担任男生班长和女生班长(男生班长必须是男生,女生班长必须是女生),则共有C(32,2)/2C(32,2)/2种不同的选法。

正确答案错误

解析详情

【答案】错误

【考点】分类计数

【解析】 男生班长和女生班长的选择应分别在男生、女生中各选1人,方案数应是“男生人数 × 女生人数”。C(32,2)/2 既没有利用性别限制,也把问题错误地当成从32人里任取两人。

【易错点】 带角色和类别限制的计数题,通常要先按类别拆开再相乘。