Logo

2024年9月 GESP C++ 6级

GESP · 6级 · 2024-09

60:00
满分 100
时长 60 分钟
27

2024年9月 GESP C++ 6级认证考试真题(含编程操作题部分)

答题卡 已答 0/27
已答 正确 错误 编程题

单选题(共 15 题,每题 2 分)

1

以下( )没有涉及 C++ 语言的面向对象特性支持。

2

关于以下 C++ 代码,( )行代码会引起编译错误。

#include <iostream>
using namespace std;
class Base {
    private:
    int a;
    protected:
    int b;
    public:
    int c;
    Base() : a(1), b(2), c(3) {}
};
class Derived : public Base {
    public:
    void show() {
        cout << a << endl; // Line 1
        cout << b << endl; // Line 2
        cout << c << endl; // Line 3
    }
};
3

有 $6$ 个元素,按照 $6,5,4,3,2,1$ 的顺序进入栈 $S$,下列( )的出栈序列是不能出现的( )。

4

采用如下代码实现检查输入的字符串括号是否匹配,横线上应填入的代码为( )。

#include <iostream>
#include <stack>
#include <string>
using namespace std;
bool is_valid(string s) {
    stack<char> st;
    char top;
    for (char& ch : s) {
        if (ch == '(' || ch == '{' || ch == '[') {
            st.push(ch); // 左括号入栈
        }
        else
        {
            if (st.empty())
                return false;
            ———————————————————————— // 在此处填入代码
            if ((ch == ')' && top != '(') ||
                (ch == '}' && top != '{') ||
                (ch == ']' && top != '[')) {
                return false;
            }
        }
    }
    return st.empty(); // 栈为空则说明所有括号匹配成功
}
5

下面代码判断队列的第一个元素是否等于 $a$,并删除该元素,横向上应填写( )。

#include <iostream>
#include <queue>
using namespace std;
bool is_front_equal(std::queue<int>& q, int a) {
    bool is_equal = false;
    if (!q.empty()) {
        ———————————————————————— // 在此处填入代码
    }
    return is_equal;
}
6

假设字母表 {$a$, $b$, $c$, $d$, $e$} 在字符串出现的频率分别为 $10%$,$15%$,$30%$,$16%$,$29%$。若使用哈夫曼编码方式对字母进行二进制编码,则字符 $a$ $b$ $c$ $d$ $e$ 分别对应的一组哈夫曼编码的长度分别为( )。

7

以下 C++ 代码实现 $n$ 位的格雷码,则横线上应填写( )。

#include <iostream>
#include <vector>
#include <string>
using namespace std;
// 生成 n 位的格雷码
vector<string> generate_graycode(int n) {
    vector<string> graycode_list;
    if (n <= 0) {
        return graycode_list;
    }
    // 初始1位格雷码
    graycode_list.push_back("0");
    graycode_list.push_back("1");
    // 迭代生成 n 位的格雷码
    for (int i = 2; i <= n; i++) {
        int current_size = graycode_list.size();
        for (int j = current_size - 1; j >= 0; j--) {
            graycode_list.push_back("1" + graycode_list[j]);
        }
        for (int j = 0; j < current_size; j++) {
            ———————————————————————— // 在此处填入代码
        }
    }
    return graycode_list;
}
8

给定一棵二叉树,其前序遍历结果为:ABDECFG,中序遍历结果为:DEBACFG,则这棵树的正确后序遍历结果是( )。

9

一棵有 $n$ 个结点的完全二叉树用数组进行存储与表示,已知根结点存储在数组的第 $1$ 个位置。若存储在数组第 $9$ 个位置的结点存在兄弟结点和两个子结点,则它的兄弟结点和右子结点的位置分别是( )。

10

二叉树的深度定义为从根结点到叶结点的最长路径上的结点数,则以下基于二叉树的深度优先搜索实现的深度计算函数中横线上应填写( )。

// 定义二叉树的结点结构
struct tree_node {
    int val;
    tree_node* left;
    tree_node* right;
    tree_node(int x) : val(x), left(nullptr), right(nullptr) {}
};
// 计算二叉树的深度
int max_depth(tree_node* root) {
    if (root == nullptr) {
        return 0; // 如果根结点为空,则深度为 0
    }
    int left_depth = max_depth(root->left);
    int right_depth = max_depth(root->right);
    ———————————————————————— // 在此处填入代码
}
11

上一题的二叉树深度计算还可以采用二叉树的广度优先搜索来实现。以下基于二叉树的广度优先搜索实现的深度计算函数中横线上应填写( )。

#include <queue>
int max_depth_bfs(tree_node* root) {
    if (root == nullptr) {
        return 0; // 如果树为空,深度为 0
    }
    queue <tree_node*> q;
    q.push(root);
    int depth = 0;
    // 使用队列进行层序遍历
    while (!q.empty()) {
        ———————————————————————— // 在此处填入代码
        for (int i = 0; i < level_size; ++i) {
            tree_node* node = q.front();
            q.pop();
            if (node->left) {
                q.push(node->left);
            }
            if (node->right) {
                q.push(node->right);
            }
        }
    }
    return depth;
}
12

二叉搜索树中的每个结点,其左子树的所有结点值都小于该结点值,右子树的所有结点值都大于该结点值。以下代码对给定的整数数组(假设数组中没有数值相等的元素),构造一个对应的二叉搜索树,横线上应填写( ):

// 定义二叉树的结点结构
struct tree_node {
    int val;
    tree_node* left;
    tree_node* right;
    tree_node(int x) : val(x), left(nullptr), right(nullptr) {}
};
// 插入结点到二叉搜索树中
tree_node* insert(tree_node* root, int val) {
    if (root == nullptr) {
        return new tree_node(val);
    }
    ———————————————————————— // 在此处填入代码
    return root;
}
// 根据给定数组构造二叉搜索树
tree_node* constructBST(const int arr[], int size) {
    tree_node* root = nullptr;
    for (int i = 0; i < size; ++i) {
        root = insert(root, arr[i]);
    }
    return root;
}
13

对上题中的二叉搜索树,当输入数组为时,构建二叉搜索树,并采用如下代码实现的遍历方式,得到的输出是( )。

#include <iostream>
using namespace std;
// 遍历二叉搜索树,输出结点值
void traversal(tree_node* root) {
    if (root == nullptr) {
        return;
    }
    traversal(root->left);
    cout << root->val << " ";
    traversal(root->right);
}
14

动态规划通常用于解决( )。

15

阅读以下用动态规划解决的 0-1 背包问题的函数,假设背包的容量是 $10$ kg,假设输入 $4$ 个物品的重量分别为(单位为 kg),每个物品对应的价值分别为,则函数的输出为( )。

#include <iostream>
#include <vector>
using namespace std;
// 0/1 背包问题
int knapsack(int W, const vector<int>& weights, const vector<int>& values, int n) {
    vector<vector<int>> dp(n + 1, vector<int>(W + 1, 0));
    for (int i = 1; i <= n; ++i) {
        for (int w = 0; w <= W; ++w) {
            if (weights[i - 1] <= w) {
                dp[i][w] = max(dp[i - 1][w], dp[i - 1][w - weights[i - 1]] + values[i - 1]);
            } else {
                dp[i][w] = dp[i - 1][w];
            }
        }
    }
    return dp[n][W];
}

判断题(共 10 题,每题 2 分)

16

C++、Python 和 JAVA 等都是面向对象的编程语言。

17

在 C++ 中,类的静态成员变量只能被该类对象的成员函数访问。

18

栈是一种线性结构,可通过数组或链表来实现。二者相比,数组实现占用的内存较少,链表实现的入队和出队操作的时间复杂度较低。

19

运行以下 C++ 代码,屏幕将输出“derived class”。

#include <iostream>
using namespace std;
class base {
    public:
    virtual void show() {
        cout << "base class" << endl;
    }
};
class derived : public base {
    public:
    void show() override {
        cout << "derived class" << endl;
    }
};
int main() {
    base* b;
    derived d;
    b = &d;
    b->show();
    return 0;
}
20

如下列代码所示的基类(base)及其派生类(derived),则生成一个派生类的对象时,只调用派生类的构造函数。

#include <iostream>
using namespace std;
class base {
    public:
    base() {
        cout << "base constructor" << endl;
    }
    ~base() {
        cout << "base destructor" << endl;
    }
};
class derived : public base {
    public:
    derived() {
        cout << "derived constructor" << endl;
    }
    ~derived() {
        cout << "derived destructor" << endl;
    }
};
21

哈夫曼编码本质上是一种贪心策略。

22

如果根结点的深度记为 $1$,则一棵恰有 $n$ 个叶结点的二叉树的深度最少是 $\lceil \log_2 n \rceil + 1$。

23

在非递归实现的树的广度优先搜索中,通常使用栈来辅助实现。

24

状态转移方程是动态规划的核心,可以通过递推方式表示问题状态的变化。

25

应用动态规划算法时,识别并存储重叠子问题的解是必须的。

编程操作题(共 2 题,共 50 分)

26
编程操作题 25分

试题名称:小杨和整数拆分

时间限制:1.0 s | 内存限制:512.0 MB

题目描述

小杨有一个正整数 $n$,小杨想将它拆分成若干完全平方数的和,同时小杨希望拆分的数量越少越好。

编程计算总和为 $n$ 的完全平方数的最小数量。

输入格式

输入只有一行一个正整数 $n$。

输出格式

输出一行一个整数表示答案。

样例输入 #1

18

样例输出 #1

2

说明/提示

数据规模与约定

对全部的测试数据,保证 $1 \leq n \leq 10^5$。

27
编程操作题 25分

试题名称:算法学习

时间限制:1.0 s | 内存限制:512.0 MB

题目描述

小杨计划学习 $m$ 种算法,为此他找了 $n$ 道题目来帮助自己学习,每道题目最多学习一次。

小杨对于 $m$ 种算法的初始掌握程度均为 $0$。第 $i$ 道题目有对应的知识点 $a_i$,即学习第 $i$ 道题目可以令小杨对第 $a_i$ 种算法的掌握程度提高 $b_i$。小杨的学习目标是对于 $m$ 种算法的掌握程度均至少为 $k$。

小杨认为连续学习两道相同知识点的题目是不好的,小杨想请你编写程序帮他计算出他最少需要学习多少道题目才能使得他在完成学习目标的同时避免连续学习两道相同知识点的题目。

输入格式

第一行三个正整数 $m, n, k$,代表算法种类数,题目数和目标掌握程度。
第二行 $n$ 个正整数 $a_1, a_2, ..., a_n$,代表每道题目的知识点。
第三行 $n$ 个正整数 $b_1, b_2, ..., b_n$,代表每道题目提升的掌握程度。

输出格式

输出一个整数,代表小杨最少需要学习题目的数量,如果不存在满足条件的方案,输出 -1。

样例输入 #1

3 5 10
1 1 2 3 3
9 1 10 10 1

样例输出 #1

4

样例输入 #2

2 4 10
1 1 1 2
1 2 7 10

样例输出 #2

-1

说明/提示

样例 1 解释

一种最优学习顺序为第一道题,第三道题,第四道题,第二道题。

数据规模与约定

| 子任务编号 | 数据点占比 | $m$ | $n$ | $b_i$ | $k$ |
| :-: | :-: | :-: | :-: | :-: | :-: |
| $1$ | $30%$ | $2$ | $\leq 9$ | $\leq 10$ | $\leq 10$ |
| $2$ | $30%$ | $\leq 9$ | $\leq 9$ | $\leq 10$ | $\leq 10$ |
| $3$ | $40%$ | $\leq 10^5$ | $\leq 10^5$ | $\leq 10^5$ | $\leq 10^5$ |

对于全部数据,保证有 $1 \leq m, n, b_i, k \leq 10^5$,$1 \leq a_i \leq m$。

已答 0/27