社区讨论

真题(终结版)

学术版参与者 6已保存回复 6

讨论操作

快速查看讨论及其快照的属性,并进行相关操作。

当前回复
6 条
当前快照
1 份
快照标识符
@mhj9faa4
此快照首次捕获于
2025/11/03 22:53
4 个月前
此快照最后确认于
2025/11/03 22:53
4 个月前
查看原帖
书接上回, luogu.me>10000字符炸了,就再发一遍。 注意要复制往下拉,代码和主题分别复制
CPP
$1$. 有 5 个红色球和 5 个蓝色球,它们除了颜色之外完全相同。将这 10 个球拍成一排,要求任意两个蓝色球都不能相邻,有多少种不同的排列方法?

$\text{A}$. 25

$\text{B}$. 30

$\text{C}$. 6

$\text{D}$. 120

$2$. 在 KMP 算法中,对于模式串 P="abacaba",其 next 数组($next_i$ 定义为模式串 $P_0$ ~ $P_i$ 最长公共前后缀的长度,且数组下标从 0 开始)的值是什么?

$\text{A}$. {0, 0, 1, 0, 1, 2, 3}

$\text{B}$. {0, 1, 2, 3, 4, 5, 6}

$\text{C}$. {0, 0, 1, 1, 2, 2, 3}

$\text{D}$. {0, 0, 0, 0, 1, 2, 3}

$3$. 对一个大小为 16(下标 0-15)的数组上构建满线段树。查询区间 $[3,11]$ 时,最少需要访问多少个树结点(包括路径上的父结点和完全包含在查询区间内的结点)?

$\text{A}$. 7

$\text{B}$. 8

$\text{C}$. 9

$\text{D}$. 10

$4$. 将字符串 "cat""car""cart""case""dog""do" 插入一个空的 Trie 树(前缀树)中。构建完成 Trie 树(包括根节点)共有多少个结点?

$\text{A}$. 8

$\text{B}$. 9

$\text{C}$. 10

$\text{D}$. 11

$5$. 对于一个包含 $n$ 个结点和 $m$ 条边的有向无环图(DAG),其拓扑排序的结果有多少种可能?

$\text{A}$. 只有 1 种

$\text{B}$. 最多 $n$ 种

$\text{C}$. 等于 $n-m$ 种

$\text{D}$. 以上都不对

$6$. 在一个大小为 13 的哈希表中,使用闭散列法的线性探查来解决冲突。哈希函数为 $H(key)=key mod 13$。依次插入关键字 18263596874。插入 74 后,它最终被放置在哪个索引位置?

$\text{A}$. 5

$\text{B}$. 7

$\text{C}$. 9

$\text{D}$. 11

$7$. 一个包含 8 个顶点的完全图(顶点的编号为 18),任意两点之间的边权重等于两顶点编号的差的绝对值。例如,顶点 37 之间的边权重为 $∣73∣=4$。该图的最小生成树总权重是多少?

$\text{A}$. 7

$\text{B}$. 8

$\text{C}$. 9

$\text{D}$. 10

$8$. 如果一棵二叉搜索树的后序遍历序列是 254812106,那么该树的前序遍历是什么?

$\text{A}$. 642510812

$\text{B}$. 645210128

$\text{C}$. 245681012

$\text{D}$. 128105246

$9$. 一个 0-1 背包问题,背包容量为 20。现有 5 个物品,其重量和价值分别为 7543615129713。装入背包的物品能获得的最大总价值是多少?

$\text{A}$. 43

$\text{B}$. 41

$\text{C}$. 45

$\text{D}$. 44

$10$. 在一棵以结点 1 为根的树中,结点 12 和结点 18 的最近公共祖先(LCA)是结点 4。 那么下列哪个结点的 LCA 组合是不可能出现的?

$\text{A}$. LCA(12,4)=4

$\text{B}$. LCA(18,4)=4

$\text{C}$. LCA(12,18,4)=4

$\text{D}$. LCA(12,1)=4

$11$. 递归关系式 $T(n)=2T(n/2)+O(n^2)$ 描述了某个分治算法的时间复杂度。 请问该算法的时间复杂度是多少?

$\text{A}$.$ O(n)$

$\text{B}$. $O(nlogn)$

$\text{C}$. $O(n^2)$

$\text{D}$. $O(n ^2logn)$

$12$. 在一个初始为空的最小堆(min-heap)中,依次插入元素 20, 12, 15, 8, 10, 5。然后连续执行两次”删除最小值”(delete-min) 操作。请问此时堆顶元素是什么?

$\text{A}$. 10

$\text{B}$. 12

$\text{C}$. 15

$\text{D}$. 20

$$13$$. 11000 之间,不能被 235 中任意一个数整除的整数有多少个?

$\text{A}$. 266

$\text{B}$. 267

$\text{C}$. 333

$\text{D}$. 734

$$14$$. 斐波那契数列的定义为 $F(0)=0$, $F(1)=1$, $F(n)=F(n−1)+F(n−2)$。使用朴素递归方法计算 $F(n)$ 的时间复杂度是指数级的。而使用动态规划(或迭代)方法的时间复杂度是线性的。造成这种巨大差异的根本原因是?

$\text{A}$. 递归函数调用栈开销过大

$\text{B}$. 操作系统对递归深度有限制

$\text{C}$. 朴素递归中存在大量的重叠子问题未被重复利用

$\text{D}$. 动态规划使用了更少的数据存储空间

$15$. 有 5 个独立的、不可抢占的任务 $\text{A1}$, $\text{A2}$, $\text{A3}$, $\text{A4}$, $\text{A5}$ 需要在一台机器上执行( 从时间 0 开始执行),每个任务都有对应的处理时长和截止时刻,按顺序分别为 3, 4, 2, 5, 15, 10, 3, 15, 11。 如果某一个任务超时,相应的惩罚等于其处理时长。为了最小化总惩罚,应该优先执行哪个任务?

$\text{A}$. 处理时间最短的任务 $\text{A5}$

$\text{B}$. 截止时间最早的任务 $\text{A3}$

$\text{C}$. 处理时间最长的任务 $\text{A4}$

$\text{D}$. 任意一个任务都可以

阅读程序:

$1$.
CPP
#include <algorithm>//暂时无法看行
#include <cstdio>
#include <cstring>
bool flag[27];
int n;
int p[27];
int ans = 0;
void dfs(int k) {
  if (k == n + 1){
    ++ ans;
    return;
  }
  for (int i = 1; i <= n; ++i) {
    if (flag[i]) continue;
    if (k > 1 && i == p[k - 1] + 1) continue;
    p[k] = i;
    flag[i] = true;
    dfs(k + 1);
    flag[i] = false;
  }
  return;
}
int main() {
  scanf("%d", &n);
  dfs(1);
  printf("%d\n", ans);
  return 0;
}
CPP
$(1)$. (1 分)当输入的 $n=3$ 的时候,程序输出的答案为 $3$。

$\text{A}$. 正确

$\text{B}$. 错误

$(2)$. 在 dfs 函数运行过程中,$k$ 的取值会满足 $1≤k≤n+1$。

$\text{A}$. 正确

$\text{B}$. 错误

$(3)$. 删除第 19 行的 “flag[i]=false;”,对答案不会产生影响。

$\text{A}$. 正确

$\text{B}$. 错误

$(4)$. 当输入的 $n=4$ 的时候, 程序输出的答案为( ) 。

$\text{A}$. 11

$\text{B}$. 12

$\text{C}$. 24

$\text{D}$. 9

$(5)$. 如果因为某些问题,导致程序运行第 25 行的 dfs 函数之前,数组 $p$ 的初值并不全为 0,则对程序的影响是( )。

$\text{A}$. 输出的答案比原答案要小

$\text{B}$. 无法确定输出的答案

$\text{C}$. 程序可能陷入死循环

$\text{D}$. 没有影响

$(6)$. 假如删去第 14 行的 “if(flag[i]) continue;” ,输入 3,得到的输出答案是( )。

$\text{A}$. 27

$\text{B}$. 3

$\text{C}$. 16

$\text{D}$. 12
CPP
#include <algorithm>
#include <cstdio>
#include <cstring>
#define ll long long
int cnt_broken = 0;
int cnt_check = 0;
int n, k;
inline bool check(int h) {
  printf("now check:%d\n", h);
  ++cnt_check;
  if (cnt_broken == 2) {
    printf("You have no egg!\n");
    return false;
  }
  if (h >= k) {
    ++cnt_broken;
    return true;
  } else {
    return false;
  }
}
inline bool assert_ans(int h) {
  if (h == k) {
    printf("You are Right using %d checks\n", cnt_check);
    return true;
  } else {
    printf("Wrong answer!\n");
    return false;
  }
}
inline void guess1(int n) {
   for (int i = 1; i <= n; ++i) {
     if (check(i)) {
       assert_ans(i);
       return;
     }
   }
}
inline void guess2(int n) {
  int w = 0;
  for (w = 1; w * (w + 1) / 2 < n; ++w)
    ;
  for (int ti = w, nh = w;; --ti, nh += ti, nh = std::min(nh, n)) {
   if (check(nh)) {
    for (int j = nh - ti + 1; j < nh; ++j) {
      if (check(j)) {
        assert_ans(j);
        return;
       }
    }
      assert_ans(nh);
      return;
     }
   }
}
int main() {
  scanf("%d%d", &n, &k);
  int t;
  scanf("%d", &t);
  if (t == 1) {
    guess1(n);
  } else {
   guess2(n);
  }
  return 0;
}
CPP

( 注意:下述的 “猜测数” 为调用 check 函数的次数(即 cnt_check 的值);“猜测正确” 的含义为 assert_ans 函数 return true(执行第 25 行所在分支)的情况;所有输入保证 1 ≤ k ≤ n。)

$(1)$. 当输入为 “6 5 1” 时,猜测次数为 5;当输入 “6 5 2” 时,猜测次数为 3。

$\text{A}$. 正确

$\text{B}$. 错误

$(2)$. 不管输入的 $n$ 和 $k$ $(2)$. 不管输入的 $n$ 和 $k$ 具体为多少,$t=2$ 时的猜测数总是小于等于$t=1$ 时的猜测数。

(3). 不管 t=1 或 t=2,程序都一定会猜到正确结果。
具体为多少,$t=2$ 时的猜测数总是小于等于$t=1$ 时的猜测数。

$\text{A}$. 正确

$\text{B}$. 错误

$(2)$. 不管输入的 $n$ 和 $k$ 具体为多少,$t=2$ 时的猜测数总是小于等于$t=1$ 时的猜测数。

$\text{A}$. 正确

$\text{B}$. 错误

$(3)$. 不管 $t=1$ 或 $t=2$,程序都一定会猜到正确结果。

$\text{A}$. 正确

$\text{B}$. 错误

$(4)$. 函数 $guess1$ 在运行过程中,cnt_broken的值最多为( ) 。

$\text{A}$. 0

$\text{B}$. 1

$\text{C}$. 2

$\text{D}$. n

$(5)$. 函数 $guess2$在运行过程中,最多使用的猜测次数的量级为( )。

$\text{A}$.$O(n)$

$\text{B}$. $O(n^2)$

$\text{C}$.$ O(n)$

$\text{D}$. $O(logn)$

$(6)$. 当输入的 $n=100$ 的时候,代码中 $t=1$ 和 $t=2$ 分别需要的猜测次数最多分别为( )。

$\text{A}$. 10014

$\text{B}$. 10013

$\text{C}$. 9914

$\text{D}$. 9913

$3$.
CPP
#include <algorithm>
#include <cstdio>
#include <cstring>
#include <vector>
#define ll long long
int n, m;
std::vector<int> k, p;
inline int mpow(int x, int k) {
   int ans = 1;
   for (; k; k = k >> 1, x = x * x) {
      if (k & 1)
       ans = ans * x;
   }
   return ans;
}
std::vector<int> ans1, ans2;
int cnt1, cnt2;
inline void dfs(std::vector<int>& ans, int& cnt, int l, int r, int v) {
   if (l > r) {
	++cnt;
	ans.push_back(v);
	return;
   }
   for (int i = 1; i <= m; ++i) {
	dfs(ans, cnt, l + 1, r, v + k[l] * mpow(i, p[l]));
   }
     return;
}
std::vector<int> cntans1;
int main() {
	scanf("%d%d", &n, &m);
	k.resize(n + 1);
	p.resize(n + 1);
	for (int i = 1; i <= n; ++i) {
		scanf("%d%d", &k[i], &p[i]);
	}
	dfs(ans1, cnt1, 1, n >> 1, 0);
	dfs(ans2, cnt2, (n >> 1) + 1, n, 0);
	std::sort(ans1.begin(), ans1.end());
	int newcnt1 = 1;
	cntans1.push_back(1);
	for (int i = 1; i < cnt1; ++i) {
    if (ans1[i] == ans1[newcnt1 - 1]) {
  			++cntans1[newcnt1 - 1];
    } else {
  		ans1[newcnt1++] = ans1[i];
      cntans1.push_back(1);
    }
	}
	cnt1 = newcnt1;
	std::sort(ans2.begin(), ans2.end());
	int las = 0;
	ll ans = 0;
	for (int i = cnt2 - 1; i >= 0; --i) {
  for (; las < cnt1 && ans1[las] + ans2[i] < 0; ++las)
			;
		if (las < cnt1 && ans1[las] + ans2[i] == 0)
		ans += cntans1[las];
	}
	printf("%lld\n", ans);
	return 0;
}
CPP
$(1)$. 删除第 51 行的“ std::sort(ans2.begin(),ans2.end());” 后, 代码输出的结果不会受到影响。

$\text{A}$. 正确

$\text{B}$. 错误

$(2)$. 假设计算过程中不发生溢出, 函数 $mpow(x,k)$的功能是求出 $x$ $k$ 的取值。 ( )

$\text{A}$. 正确

$\text{B}$. 错误$

$(3)$. 代码中第 39 行到第 50 行的目的是为了将 $ans1$ 数组进行“ 去重” 操作。 ( )

$\text{A}$. 正确

$\text{B}$. 错误

(4). 当输入为“ 3 15 1 2 -1 2 1 2” 时, 输出结果为( )

$\text{A}$. 4

$\text{B}$. 8

$\text{C}$. 0

$\text{D}$. 10

(5). 记程序结束前 $p$ 数组元素的最大值为 $P$, 则该代码的时间复杂度是( )

$\text{A}$. $O(n)$

$\text{B}$. $O(mn(logm)n)$

$\text{C}$. $O(m⋅n/2⋅logm⋅n/2)$

$\text{D}$. $O(m⋅n/2⋅(logm⋅n/2+logP))$

(6). 本题所求出的是( ) 。

$\text{A}$. 满足 $a,b,c∈[1,m]$ 的整数方程 $a ^3+b^ 3=c^3$
 的解的数量

$\text{B}$. 满足 a,b,c∈[1,m] 的整数方程 $a ^2+b^ 2=c^2$
  的解的数量

$\text{C}$. 满足 $x_i∈[0,m]$ 的整数方程 $\sum_{i=1}^{n} x_i^{p_i}$
 =0 的解的数量

$\text{D}$. 满足 $x_i∈[1,m]$ 的整数方程 $\sum_{i=1}^{n} x_i^{p_i}$
 =0 的解的数量

完善程序:

1. (特殊最短路)给定一个含 N 个点、M 条边的带权无向图,边权非负。起点为 S,终点为 T。对于一条 S 到 T 的路径,可以在整条路径中,至多选择一条边作为“免费边”:当第一次经过这条被选中的边时,费用视为 0;如果之后再次经过该边,则仍按其原始权重计费。点和边均允许重复经过。求从 S 到 T 的最小总费用。

以下代码求解了上述问题。试补全程序。

CPP
#include <algorithm>
#include <iostream>
#include <queue>
#include <vector>
using namespace std;

const long long INF = 1e18;

struct Edge {
    int to;
    int weight;
};

struct State {
    long long dist;
    int u;
    int used_freebie;
    bool operator>(const State &other) const {
        return dist > other.dist;
    }
};

int main() {
    int n, m, s, t;
    cin >> n >> m >> s >> t;

    vector<vector<Edge>> adj(n + 1);
    for (int i = 0; i < m; ++i) {
        int u, v, w;
        cin >> u >> v >> w;
        adj[u].push_back({v, w});
        adj[v].push_back({u, w});
    }

    vector<vector<long long>> d(n + 1, vector<long long>(2, INF));
    priority_queue<State, vector<State>, greater<State>> pq;

    d[s][0] = 0;
    pq.push({0, s, ①});

    while (!pq.empty()) {
        State current = pq.top();
        pq.pop();

        long long dist = current.dist;
        int u = current.u;
        int used = current.used_freebie;

        if (dist > ②) {
            continue;
        }

        for (const auto &edge : adj[u]) {
            int v = edge.to;
            int w = edge.weight;

            if (d[u][used] + w < ③) {
                ③ = d[u][used] + w;
                pq.push({③, v, used});
            }

            if (used == 0) {
                if (④ + w < d[v][1]) {
                    d[v][1] = ④ + w;
                    pq.push({d[v][1], v, 1});
                }
            }
        }
    }

    cout << ⑤ << endl;
    return 0;
}
CPP
$(1)$. ①处应填( )

$\text{A}$. 0

$\text{B}$. 1

$\text{C}$. -1

$\text{D}$. false

$(2)$. ②处应填( )

$\text{A}$. d[u][!used]

$\text{B}$. d[u][used]

$\text{C}$. d[t][used]

$\text{D}$. INF

$(3)$. ③处应填( )

$\text{A}$. d[v][1]

$\text{B}$. d[v][used]

$\text{C}$. d[u][used]

$\text{D}$. d[v][0]

$(4)$. ④处应填( )

$\text{A}$. d[v][0]

$\text{B}$. d[v][1]

$\text{C}$. d[u][0]

$\text{D}$. d[u][1]

$(5)$. ⑤处应填( )

$\text{A}$. d[t][1]

$\text{B}$. d[t][0]

$\text{C}$. min(d[t][0], d[t][1])

$\text{D}$. d[t][0] + d[t][1]

2.

工厂打算通过客户反馈来间接测试生产线,从而找到存在缺陷的生产线。工厂有 n 条生产线(编号 0∼n−1),已知其中恰有一条生产线存在缺陷。每一轮测试为,从若干生产线的产品取样混合成一个批次发给客户。若该批次中包含缺陷生产线的产品,客户将要求退货(结果记为 1),否则正常收货(记为 0)。受售后压力限制,在所有发货批次中,最多只能有 k 次退货(即结果为 1 的次数 ≤k)。工厂的目标是,设计最少的间接测试轮数 w(发货总批次),保证根据客户收货或退货的反馈结果,唯一确定存在缺陷的生产线。

以下程序实现了工厂的目标,包含两部分:i) 确定 w 的最小值,并设计最优测试方案;ii) 根据测试结果推断存在缺陷的生产线。该程序确定 w 最小值的方法为:由于不同的生产线故障时,测试应当返回不同的结果,因此 w 轮测试的可能结果数不应少于生产线数量。

test_subset() 函数为抽象测试接口,输入所有批次的方案并返回一个二进制编码;该编码表示为每批次的检测结果(即最低位是第 1 批次、最高位是第 w 批次);其实现在此处未给出。

test_subset() 函数为抽象测试接口,输入所有批次的方案并返回一个二进制编码;该编码表示为每批次的检测结果(即最低位是第 1 批次、最高位是第 w 批次);其实现在此处未给出。

试补全程序。
CPP
#include <algorithm>
#include <cstddef>
#include <iostream>
#include <vector>
using namespace std;

long long comb(int w, int i) {
    if (i < 0 || i > w) {
        return 0;
    }
    long long res = 1;
    for (int t = 1; t <= i; ++t) {
        res = res * (w - t + 1) / t;
    }
    return res;
}

long long count_patterns(int w, int k) {
    long long total = 0;
    for (int t = 0; t <= min(w, k); ++t) {
        total += comb(w, t);
    }
    return total;
}

int test_subset(const vector<vector<int>> &plan);

int solve(int n, int k) {
    int w = 1;
    while (①) {
        ++w;
    }
    cout << w << endl;

    vector<vector<int>> code(n, vector<int>(w, 0));
    int idx = 0;
    for (int ones = 0; ones <= k && idx < n; ++ones) {
        vector<int> bits(w, 0);
        fill(bits.begin(), bits.begin() + ones, 1);
        do {
            for (int b = 0; b < w; ++b) {
                code[idx][b] = bits[b];
            }
            ++idx;
            if (idx >= n) {
                break;
            }
        } while (②);
    }

    vector<vector<int>> plan(w);
    for (int i = 0; i < w; ++i) {
        for (int j = 0; j < n; ++j) {
            if (③) {
                plan[i].push_back(j);
            }
        }
    }

    int signature = test_subset(plan);

    vector<int> sig_bits(w, 0);
    for (int i = 0; i < w; ++i) {
        if (④) {
            sig_bits[i] = 1;
        }
    }

    for (int j = 0; j < n; ++j) {
        if (⑤) return j;
    }
}

int main() {
    int n,k;
    cin >> n >> k;
    int ans = solve(n, k);
    cout << ans << endl;
    return 0;
}
CPP
$(1)$. ①处应填 ( )

$\text{A}$. (1 << w) < n

$\text{B}$. count_patterns(w, k) < n

$\text{C}$. count_patterns(k, w) < n

$\text{D}$. comb(w, k) < n

$(2)$. ②处应填 ( )

$\text{A}$. next_permutation(bits.begin(), bits.end())

$\text{B}$. prev_permutation(bits.begin(), bits.end())

$\text{C}$. next_permutation(bits.begin(), bits.begin()+ones)

$\text{D}$. prev_permutation(bits.begin(), bits.begin()+ones)


$(3)$. ③处应填 ( )


$\text{A}$. (j >> i) & 1

$\text{B}$. (i >> j) & 1

$\text{C}$. code[i][j] == 1

$\text{D}$. code[j][i] == 1

$(4)$. ④处应填 ( )

$\text{A}$. (signature >> i) & 1

$\text{B}$. (signature >> i) ^ 1

$\text{C}$. signature | (1 << i)

$\text{D}$. (signature >> i) | 1

$(5)$. ⑤处应填 ( )

$\text{A}$. is_permutation(code[j].begin(), code[j].end(), sig_bits.begin())

$\text{B}$. code[j] == sig_bits

$\text{C}$. plan[j] == sig_bits

$\text{D}$. code[j][i] == sig_bits[i]
这样就不炸了

回复

6 条回复,欢迎继续交流。

正在加载回复...