#GESP202603C5T1. 选择题
选择题
第 1 题
关于单链表、双链表和循环链表,下列说法正确的是()。 [cite: 1559]
{{ select(1) }}
- 在单链表中,若已知任意结点的指针,则可以在O(1)时间内删除该结点。
- 循环链表中一定不存在空指针。
- 在循环双链表中,尾结点的 next 指针一定为nullptr。
- 在带头结点的循环单链表中,判定链表是否为空只需判断头结点的 next 是否指向自身。
第 2 题
双向循环链表中要在结点p之前插入新结点s(均非空),以下指针操作正确的是()。 [cite: 1565]
{{ select(2) }}
s->next = p; p->prev = s; q->next = s; s->prev = q;s->prev = p; s->next = p->next; p->next->prev = s; p->next = s;s->next = p; s->prev = p->prev; p->prev->next = s; p->prev = s;s->next = p; s->prev = nullptr; p->prev = s;
第 3 题
下面函数用“哑结点”统一处理删除单向链表中的头结点与中间结点。横线处应填()。 [cite: 1601]
Node* eraseAll (Node* head, int x) {
Node dummy (0);
dummy.next = head;
Node* cur = &dummy;
while (cur->next){
if (cur->next->val == x) {
Node* del = cur->next;
____________________;
delete del;
} else cur = cur->next;
}
return dummy.next;
}
{{ select(3) }}
cur = cur->next;cur->next = del->next;del->next = cur->next;cur->next = nullptr;
第 4 题
对如下代码实现的欧几里得算法(辗转相除法),执行 gcd(48,18) 得到的调用序列为()。 [cite: 1652]
int gcd(int a, int b) {
return b == 0 ? a : gcd(b, a % b);
}
{{ select(4) }}
gcd(48,18) -> gcd(18,12) -> gcd(12,6) -> gcd(6,0)gcd(48,18) -> gcd(30,18) -> gcd(12,18)gcd(48,18) -> gcd(18,30) -> gcd(30,6)gcd(48,18) -> gcd(12,18) -> gcd(6,12)
第 5 题
下面代码实现了欧拉(线性)筛,横线处应填写()。 [cite: 1668]
vector<int> euler_sieve (int n) {
vector<bool> is_composite (n+1, false);
vector<int> primes;
for (int i=2; i<=n; i++) {
if (!is_composite[i])
primes.push_back(i);
for (int j=0; __________ && (long long)i * primes[j] <= n; j++) {
is_composite[i * primes[j]] = true;
if (i % primes[j] == 0)
break;
}
}
return primes;
}
{{ select(5) }}
j <= nj < sqrt(n)j < primes.size()j < i
第 6 题
埃氏筛中将内层循环从 j=i*i 开始而不是 j=2*i 的主要原因是()。 [cite: 1710]
{{ select(6) }}
- 因为2*i一定不是合数
- i*i一定是质数
- 小于i*i的i的倍数已被更小质因子筛过
- 这样可以把时间复杂度降为O(n)
第 7 题
下面程序的运行结果为()。 [cite: 1738]
// 包含二分查找答案与 check 函数判定...
int a[] = {1, 2, 8, 4, 9};
int n = 5;
int k = 3;
std::cout << solve(n, a, k) << std::endl;
{{ select(7) }}
- 2
- 3
- 4
- 5
第 8 题
在升序数组中查找第一个大于等于x的位置,下面循环中横线应填()。 [cite: 1820]
int lower_Bound (const vector<int>& a, int x) {
int l = 0, r = a.size();
while(l < r) {
int mid = l + (r - l) / 2;
if(a[mid] >= x)
_________
else
l = mid + 1;
}
return l;
}
{{ select(8) }}
r = mid;r = mid - 1;l = mid;l = mid + 1;
第 9 题
关于递归函数调用,下列说法错误的是()。 [cite: 1842]
{{ select(9) }}
- 递归调用层次过深时,可能会耗尽栈空间导致栈溢出
- 尾递归函数可以通过编译器优化来避免栈溢出
- 所有递归函数都可以通过循环结构来改写,从而避免栈溢出
- 栈溢出发生时,程序会抛出异常并可以继续执行后续代码
第 10 题
给定n根木头,第i根长度为a[i]。要切成不少于m段等长木段,求最大可能长度,则横线上应填写()。 [cite: 1849]
// ... 二分答案模板
if (check(mid)) {
ans = mid;
_________
} else {
_________
}
// ...
{{ select(10) }}
l = mid + 1;和r = mid - 1;l = mid - 1;和r = mid + 1;l = mid + 1;和r = mid;l = mid;和r = mid + 1;
第 11 题
下面代码用分治求“最大连续子段和”,其时间复杂度为()。 [cite: 1939]
int solve (vector<int>& a, int l, int r){
if(l == r) return a[l];
int mid = l + (r - l) / 2;
int left = solve(a, l, mid);
int right = solve(a, mid + 1, r);
// ... 跨越 mid 的线性扫描 ...
return max({left, right, lmax + rmax});
}
{{ select(11) }}
- O(n²)
- O(n log n)
- O(log n)
- O(n)
第 12 题
游戏大赛决赛,两组选手分别按得分从小到大排好队,现在要把他们合并成一个有序排行榜。横线处应填入()。 [cite: 1991]
while (i < A.size() && j < B.size()) {
if(__________) {
result.push_back(A[i++]);
} else {
result.push_back(B[j++]);
}
}
{{ select(12) }}
A[i] >= B[j]A[i] <= B[j]i >= ji <= j
第 13 题
有位同学的成绩已经从小到大排好序,现在对它执行下面这段以第一个元素为 pivot的快速排序,请问此次排序的时间复杂度是()。 [cite: 2026]
// 快速排序核心代码,选择 a[l] 为 pivot
{{ select(13) }}
- O(n)
- O(n log n)
- O(n²)
- O(log n)
第 14 题
下面关于排序算法的描述中,不正确的是()。 [cite: 2061]
{{ select(14) }}
- 冒泡排序和插入排序都是稳定的排序算法
- 快速排序和归并排序都是不稳定的排序算法
- 冒泡排序和插入排序最好时间复杂度均为O(n)
- 归并排序在最好、最坏和平均三种情况的时间复杂度均为O(n log n)
第 15 题
下面代码实现两个整数除法,其中被除数为一个“大整数”,用字符串表示,除数是一个小整数,用int表示,则横线处应该填写()。 [cite: 2067]
for(int i=0; i < a.size(); i++){
rem = rem * 10 + a[i];
int q = rem / b;
c.push_back(q);
__________;
}
{{ select(15) }}
rem /= b;rem %= b;rem = b;rem = q;