Logo

2024年12月 GESP C++ 5级

GESP · 5级 · 2024-12

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

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

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

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

1

下面关于链表和数组的描述,错误的是( )。

2

在循环单链表中,节点的 next 指针指向下一个节点,最后一个节点的 next 指针指向( )。

3

为了方便链表的增删操作,一些算法生成一个虚拟头节点,方便统一删除头节点和其他节点。下面代码实现
了删除链表中值为val的节点,横线上应填的最佳代码是( )。

struct LinkedNode {
    int val;
    LinkedNode* next;
    LinkedNode(int val):val(val), next(nullptr){}
};
void removeElements(LinkedNode* head, int val) {
    if (head == nullptr) {
        return;
    }
    LinkedNode* cur;
    LinkedNode* dummyHead = new LinkedNode(0); //虚拟头节点
    ________________________________ // 在此处填入代码
    while(cur ->next != nullptr) {
        if(cur->next->val == val) {
            LinkedNode* tmp = cur->next;
            cur->next = cur->next->next;
            delete tmp;
            tmp = nullptr;
        }
        else {
            cur = cur ->next;
        }
    }
    head = dummyHead->next;
    delete dummyHead;
    dummyHead = nullptr;
}
4

对下面两个函数,说法错误的是( )。

int fibA(int n) {
    if (n <= 1) return n;
    int f1 = 0, f2 = 1;
    for (int i = 2; i <= n; ++i) {
        int temp = f2;
        f2 = f1 + f2;
        f1 = temp;
    }
    return f2;
}
int fibB(int n) {
    if (n <= 1) return n;
    return fibB(n - 1) + fibB(n - 2);
}
5

两块长方形⼟地的长宽分别为 和 米,要将它们分成正方形的小块,使得正方形的尺⼨尽可能大。小杨
采用如下的辗转相除函数gcd(24, 36)来求正方形分块的边长,则函数gcd调用顺序为( )。

int gcd(int a, int b) {
    int big = a > b ? a : b;
    int small = a < b ? a : b;
    if (big % small == 0) {
        return small;
    }
    return gcd(small, big % small);
}
6

唯一分解定理表明,每个大于1的自然数可以唯一地写成若⼲个质数的乘积。下面函数将自然数 的所有质因
数找出来,横线上能填写的最佳代码是( )。

#include <vector>
vector<int> get_prime_factors(int n) {
    vector<int> factors;
    if (n <= 1) {
        cout << "输入的数必须是大于1的正整数" << endl;
        return;
    }
    while (n % 2 == 0) {
        factors.push_back(2);
        n /= 2;
    }
    ________________________________ { // 在此处填入代码
        while (n % i == 0) {
            factors.push_back(i);
            n /= i;
        }
    }
    if (n > 2) {
        factors.push_back(n);
    }
    return factors;
}
7

下述代码实现素数表的埃拉托色尼(埃⽒)筛法,筛选出所有小于等于 的素数。

vector<int> sieve_Eratosthenes(int n) {
    vector<bool> is_prime(n +1, true);
    vector<int> primes;
    for (int i = 2; i * i <= n; i++) {
        if (is_prime[i]) {
            primes.push_back(i);
            for (int j = i * i; j <= n; j += i) {
                is_prime[j] = false;
            }
        }
    }
    for (int i = sqrt(n) + 1; i <= n; i++) {
        if (is_prime[i]) {
            primes.push_back(i);
        }
    }
    return primes;
}

下面说法,正确的是( )。

8

下述代码实现素数表的线性筛法,筛选出所有小于等于 的素数。下面说法正确的是( )。

vector<int> sieve_linear(int n) {
    vector<bool> is_prime(n +1, true);
    vector<int> primes;
    for (int i = 2; i <= n/2; i++) {
        if (is_prime[i])
        primes.push_back(i);
        for (int j = 0; j < primes.size() && i * primes[j] <= n; j++) {
            is_prime[ i * primes[j] ] = 0;
            if (i % primes[j] == 0)
            break;
        }
    }
    for (int i = n/2 +1; i <= n; i++) {
        if (is_prime[i])
        primes.push_back(i);
    }
    return primes;
}
9

考虑以下C++代码实现的快速排序算法:

int partition(vector<int>& arr, int left, int right) {
    int pivot = arr[right]; // 基准值
    int i = left - 1;
    for (int j = left; j < right; j++) {
        if (arr[j] < pivot) {
            i++;
            swap(arr[i], arr[j]);
        }
    }
    swap(arr[i + 1], arr[right]);
    return i + 1;
}
// 快速排序
void quickSort(vector<int>& arr, int left, int right) {
    if (left < right) {
        int pi = partition(arr, left, right);
        quickSort(arr, left, pi - 1);
        quickSort(arr, pi + 1, right);
    }
}

以下关于快速排序的说法,正确的是( )。

10

下面关于归并排序,描述正确的是( )。

11

给定一个长度为 的有序数组nums,其中所有元素都是唯一的。下面的函数返回数组中元素target的索
引。

int binarySearch(vector<int> &nums, int target, int left, int right) {
    if (left > right) {
        return -1;
    }
    int middle = left + ((right - left) / 2);
    if (nums[middle] == target) {
        return middle;
    }
    else if (nums[middle] < target) {
        return binarySearch(nums, target, middle + 1, right);
    }
    else
    return binarySearch(nums, target, left, middle - 1);
}
}
int Find(vector<int> &nums, int target) {
    int n = nums.size();
    return binarySearch(nums, target, 0, n - 1);
}

关于上述函数,描述不正确的是( )。

12

给定一个长度为 的有序数组nums,其中可能包含重复元素。下面的函数返回数组中某个元素target的
左边界,若数组中不包含该元素,则返回−1。例如在数组nums = [5,7,7,8,8,10]中查找target=8,函数返
回 在数组中的左边界的索引为 。则横线上应填写的代码为( )。

int getLeftBoundary(vector<int>& nums, int target) {
    int left = 0;
    int right = nums.size() - 1;
    while (left < right) {
        int middle = left + ((right - left) / 2);
        if (target <= nums[middle])
        ________________________________ // 在此处填入代码
        else
        left = middle+1;
    }
    return nums[left]==target?left:-1;
}
13

假设有多个孩⼦,数组g保存所有孩⼦的胃⼝值。有多块饼⼲,数组s保存所有饼⼲的尺⼨。小杨给孩⼦
们发饼⼲,每个孩⼦最多只能给一块饼⼲。饼⼲的尺⼨大于等于孩⼦的胃⼝时,孩⼦才能得到满足。小杨的目标是
尽可能满足越多数量的孩⼦,因此打算采用贪心算法来找出能满足的孩⼦的数目,则横线上应填写的代码为( )。

int cooki4children(vector<int>& g, vector<int>& s) {
    sort(g.begin(), g.end());
    sort(s.begin(), s.end());
    int index = s.size() - 1; // 饼干数组下标
    int result = 0;
    for (int i = g.size() - 1; i >= 0; i--) {
        if (index >= 0 && s[index] >= g[i]) {
            ________________________________ // 在此处填入代码
        }
    }
    return result;
}
14

关于分治算法,以下说法中不正确的是( )。

15

小杨编写了一个如下的高精度减法函数:

vector<int> highPrecisionSubtract(vector<int> a, vector<int> b) {
    vector<int> result;
    int borrow = 0;
    for (int i = 0; i < a.size(); ++i) {
        int digitA = a[i];
        int digitB = i < b.size() ? b[i] : 0;
        int diff = digitA - digitB - borrow;
        if (diff < 0) {
            diff += 10;
            borrow = 1;
        }
        else {
            borrow = 0;
        }
        result.push_back(diff);
    }
    while (result.size() > 1 && result.back() == 0) {
        result.pop_back();
    }
    return result;
}

下面说法,正确的是( )。

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

16

单链表只支持在表头进行插入和删除操作。

17

线性筛相对于埃拉托斯特尼筛法,每个合数只会被它的最小质因数筛去一次,因此效率更高。

18

任何一个大于1的自然数都可以分解成若⼲个不同的质数的乘积,且分解方式是唯一的。

19

贪心算法通过每一步选择当前最优解,从⽽一定能获得全局最优解。

20

递归算法必须有一个明确的结束条件,否则会导致无限递归并可能引发栈溢出。

21

快速排序和归并排序的平均时间复杂度均为 ,且都是稳定排序。

22

快速排序的时间复杂度总比插入排序的时间复杂度低。

23

二分查找仅适用于数组⽽不适合链表,因为二分查找需要跳跃式访问元素,链表中执行跳跃式访问的效率
低。

24

对有序数组{5,13,19,21,37,56,64,75,88,92,100} 进行二分查找,成功查找元素19 的比较次数是2。

25

递归函数每次调用自身时,系统都会为新开启的函数分配内存,以存储局部变量、调用地址和其他信息
等,导致递归通常比迭代更加耗费内存空间。

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

26
编程操作题 25分

试题名称:奇妙数字

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

输入格式

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

输出格式

输出⼀个正整数,代表满⾜条件的集合最多包含的奇妙数字个数。

样例解释

关于本样例,符合题意的⼀个包含 个奇妙数字的集合是 。⾸先,因为 , , ,所以 , ,
均为奇妙数字。同时, 是 的因⼦。
由于⽆法找到符合题意且同时包含 个奇妙数字的集合,因此本样例的答案为 。
子任务编号 数据点占比
1 20%
2 20%
3 60%
对于全部数据,保证有 。

27
编程操作题 25分

试题名称:武器强化

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

输入格式

第⼀⾏包含两个正整数 ,含义如题⾯所⽰。
之后 ⾏,每⾏包含两个正整数 ,代表第 种强化材料的适配武器和修改花费。

输出格式

输出⼀个整数,代表能够使适配第 种武器的强化材料种类数严格⼤于其他的武器最少需要花费的⾦币。

样例解释

花费 ,将第三种强化材料的适配武器由 改为 。此时,武器 有 种强化材料适配,武器 和武器 都各有 种
强化材料适配。满⾜适配第 种武器的强化材料种类数严格⼤于其他的武器。
子任务编号 数据点占比
1 20%
2 20%
3 60%
对于全部数据,保证有 。

已答 0/27