A.X Axis(暴力)
题意:
在 X X X轴( 1 ≤ x i ≤ 10 1\leq x_i\leq 10 1≤xi≤10)上有三个点,其整数坐标分别为 x 1 x_1 x1、 x 2 x_2 x2和 x 3 x_3 x3。您可以选择 X X X轴上任何一个整数坐标为 a a a的点。请注意,点 a a a可能与 x 1 x_1 x1、 x 2 x_2 x2或 x 3 x_3 x3重合。设 f ( a ) f(a) f(a)是给定点到点 a a a的总距离。求 f ( a ) f(a) f(a)的最小值。
点 a a a与 b b b之间的距离等于 ∣ a − b ∣ |a-b| ∣a−b∣。例如,点 a = 5 a=5 a=5与 b = 2 b=2 b=2之间的距离为 3 3 3。
分析:
暴力枚举一下所有可能,取最小值即可。
代码:
#include<bits/stdc++.h>
typedef long long LL;
using namespace std;
void solve() {
LL a, b, c;
cin >> a >> b >> c;
LL ans = 100;
for (LL i = 0; i <= 10; i++) {
ans = min(ans, abs(i - a) + abs(i - b) + abs(i - c));
}
cout << ans << endl;
}
int main() {
int t;
cin >> t;
while (t--)
solve();
return 0;
}
B.Matrix Stabilization(模拟)
题意:
给你一个大小为 n × m n\times m n×m的矩阵,其中行的编号从上到下为 1 1 1至 n n n,列的编号从左到右为 1 1 1至 m m m。位于第 i i i行和第 j j j列交点上的元素用 a i j a_{ij} aij表示。
考虑稳定矩阵 a a a的算法:
- 找到单元格 ( i , j ) (i,j) (i,j)使其值严格大于所有相邻单元格的值。如果没有这样的单元格,则终止算法。如果有多个这样的单元格,则选择 i i i值最小的单元格;如果仍有多个单元格,则选择 j j j值最小的单元格。
- 设置 a i j = a i j − 1 a_{ij}=a_{ij}-1 aij=aij−1。
- 转到步骤 1 1 1。
在此问题中,如果单元格 ( a , b ) (a,b) (a,b)和 ( c , d ) (c,d) (c,d)有一个共同的边,即 ∣ a − c ∣ + ∣ b − d ∣ = 1 |a-c|+|b-d|=1 ∣a−c∣+∣b−d∣=1,那么这两个单元格就被视为相邻单元格。
您的任务是在执行稳定算法后输出矩阵 a a a。可以证明这种算法不可能运行无限次迭代。
分析:
按题意进行模拟。从上到下,从左到右枚举元素,每次如果此元素严格大于周围的元素,就赋值为周围四个元素中最大的。因为操作以后还是没有严格小于周围的元素,所以操作后不会使得前面的元素不符合要求。
注意,边上的元素取最大值时可能会超出边界,所以每组数据要把边界外的元素都设为 0 0 0。
代码:
#include<bits/stdc++.h>
typedef long long LL;
using namespace std;
const int N = 2005;
LL a[N][N];
void solve() {
LL n, m;
cin >> n >> m;
for (LL i = 0; i <= n + 1; i++) {
a[i][0] = 0;
a[i][m + 1] = 0;
}
for (LL i = 0; i <= m + 1; i++) {
a[0][i] = 0;
a[n + 1][i] = 0;
}
for (LL i = 1; i <= n; i++) {
for (LL j = 1; j <= m; j++) {
cin >> a[i][j];
}
}
for (LL i = 1; i <= n; i++) {
for (LL j = 1; j <= m; j++) {
a[i][j] = min(a[i][j], max(a[i - 1][j], max(a[i + 1][j], max(a[i][j - 1], a[i][j + 1]))));
cout << a[i][j] << ' ';
}
cout << endl;
}
}
int main() {
int t;
cin >> t;
while (t--)
solve();
return 0;
}
C.Update Queries(排序)
题意:
下面是一个简单的问题。给你一个长度为 n n n的字符串 s s s(由小写拉丁字母组成),以及一个长度为 m m m( 1 ≤ i n d i ≤ n 1\leq ind_i\leq n 1≤indi≤n)的索引数组 i n d ind ind和一个长度为 m m m的字符串 c c c(由小写拉丁字母组成)。然后,按顺序执行更新操作,即在第 i i i个操作中,设置 s i n d i = c i s_{ind_i}=c_i sindi=ci。请注意,所有的 m m m操作都是从第一个操作到最后一个操作。
当然,如果改变数组 i n d ind ind中索引的顺序和/或字符串 c c c中字母的顺序,会得到不同的结果。如果数组 i n d ind ind中的索引和字符串 c c c中的字母顺序可以随意改变,那么在进行 m m m次更新操作后,可以得到的字典序最小的字符串 s s s。
当且仅当满足以下条件之一时,字符串 a a a在字典序上小于字符串 b b b:
- a a a是 b b b的前缀,但 a ≠ b a\neq b a=b;
- 在 a a a和 b b b不同的第一个位置,字符串 a a a中的符号在字母表中的位置比字符串 b b b中的相应符号早。
分析:
我们发现 i d n idn idn和 c c c都可以重新排列,考虑对其进行排序。如果 i d n idn idn的元素全部满足两两不相同,则直接在排序后按顺序填入即可,因为这样保证字典序小的排在了前面。对于相同的元素,先插入的元素会被后面的覆盖掉,只需要考虑最后一次操作,而前面的操作则用字典序最大的来顶替即可。
代码:
#include<bits/stdc++.h>
typedef long long LL;
using namespace std;
const int N = 1e5 + 5;
int idn[N];
string s, t;
void solve() {
LL n, m;
t = "";
cin >> n >> m >> s;
for (int i = 1; i <= m; i++)
cin >> idn[i];
for (int i = 1; i <= m; i++) {
char tmp;
cin >> tmp;
t += tmp;
}
sort(idn + 1, idn + 1 + m);
sort(t.begin(), t.end());
deque<char> q;
for (int i = 0; i < t.size(); i++)
q.push_back(t[i]);
for (int i = 1; i <= m; i++) {
if (idn[i] != idn[i - 1]) {
s[idn[i] - 1] = q.front();
q.pop_front();
} else {
q.pop_back();
}
}
cout << s << endl;
}
int main() {
int T;
cin >> T;
while (T--)
solve();
return 0;
}
D.Mathematical Problem(数学)
题意:
给你一个长度 n > 1 n\gt 1 n>1的字符串 s s s,由从 0 0 0到 9 9 9的数字组成。您必须在这个字符串中准确插入 n − 2 n-2 n−2个 + + +(加法)符号或 × \times ×(乘法)符号,才能形成一个有效的算术表达式。
在本题中,符号不能放在字符串 s s s的第一个字符之前或最后一个字符之后,也不能连续写入两个符号。此外,请注意字符串中数字的顺序不能改变。以 s = 987009 s=987009 s=987009为例:
- 从这个字符串可以得到以下算术表达式: 9 × 8 + 70 × 0 + 9 = 81 9\times 8+70\times 0+9=81 9×8+70×0+9=81, 98 × 7 × 0 + 0 × 9 = 0 98\times 7\times 0+0\times 9=0 98×7×0+0×9=0, 9 + 8 + 7 + 0 + 09 = 9 + 8 + 7 + 0 + 9 = 33 9+8+7+0+09=9+8+7+0+9=33 9+8+7+0+09=9+8+7+0+9=33。请注意,数字 09 09 09被认为是有效的,并被转换为 9 9 9。
- 从该字符串中无法得到以下算术表达式: + 9 × 8 × 70 + 09 +9\times 8\times 70+09 +9×8×70+09(符号只能放在数字之间)、 98 × 70 + 0 + 9 98\times 70+0+9 98×70+0+9(因为有 6 6 6个数字,所以必须正好有 4 4 4个符号)。
算术表达式的结果是根据数学规则计算出来的–首先进行所有乘法运算,然后是加法运算。您需要找出在给定的字符串 s s s中插入恰好 n − 2 n-2 n−2个加法或乘法符号所能得到的最小结果。
分析:
本题我们分情况考虑
考虑 n = 2 n=2 n=2的情况,不能加运算符,如果第一个字符为 0 0 0,则输出第二个字符,否则输出 s s s。
考虑 n = 3 n=3 n=3且有 0 0 0, 0 0 0在中间的情况,答案为第一个和第三个字符乘积或者相加中较小的一个。
其他长度的情况:如果有
0
0
0,则将所有运算符置为乘,最后结果就是
0
0
0。如果没有
0
0
0,则将
s
s
s中所有不为
1
1
1的数统计求和。此时的求和有
n
−
1
n-1
n−1个运算符,其中涉及到
1
1
1的运算符全为*
,不涉及
1
1
1的运算符全为+
。此时需要去掉一个运算符,再次遍历
s
s
s,依此考虑将
s
s
s中任意两个相邻的字符组合成一个二位数字,看哪个组合最后得到的结果最小,取最小值即可。
代码:
#include<bits/stdc++.h>
typedef long long LL;
using namespace std;
const LL MAXN = 0x3f3f3f3f;
string s;
void solve() {
int n;
cin >> n >> s;
if (n == 2) {
if (s[0] == '0')
cout << s[1] << endl;
else
cout << s << endl;
} else if (n == 3 && s[1] == '0') {
cout << min(s[0] + s[2] - '0' * 2, (s[0] - '0') * (s[2] - '0')) << endl;
} else if (s.find('0') != std::string::npos) {
cout << "0" << endl;
} else {
int sum = 0;
for (const auto &x: s) {
if (x > '1') {
sum += x - '0';
}
}
int ans = MAXN;
for (int i = 0; i < n - 1; ++i) {
int num = stoi(s.substr(i, 2));
ans = min(ans, sum - (s[i] == '1' ? 0 : s[i] - '0') - (s[i + 1] == '1' ? 0 : s[i + 1] - '0') + num);
}
cout << ans << endl;
}
}
int main() {
int t;
cin >> t;
while (t--)
solve();
return 0;
}
E.Beautiful Array(数据结构)
题意:
给你一个整数数组 a 1 , a 2 , … , a n a_1,a_2,\ldots,a_n a1,a2,…,an和一个整数 k k k。您需要用最少的运算使数组变得美丽。
在进行操作前,可以随意对数组元素进行洗牌。对于一次操作,可以进行以下操作:
- 选择一个索引 1 ≤ i ≤ n 1\leq i\leq n 1≤i≤n.
- 生成 a i = a i + k a_i=a_i+k ai=ai+k。
如果所有的 1 ≤ i ≤ n 1\leq i\leq n 1≤i≤n都是 b i = b n − i + 1 b_i=b_{n-i+1} bi=bn−i+1,那么数组 b 1 , b 2 , … , b n b_1,b_2,\ldots,b_n b1,b2,…,bn是美丽的。
求使数组美丽所需的最小运算次数,或者输出不可能。
分析:
首先利用map
将
a
a
a中的出现偶数次的元素去掉。我们发现如果一个数能通过另一个数相加若干个
k
k
k得到,则这两个数%k的余数是相等的。将剩下的元素
%
k
\%k
%k存到map<int,vector>
中,遍历这个新的map
,对vector
排序。如果vector
中的元素数量是偶数,那么可以将相邻的两个vector
作为一组进行
+
k
+k
+k变换。如果vector
中的元素数量是奇数,那么这样的vector
不能超过
1
1
1个,维护一个前缀和后缀和来讨论vector
中哪个元素作为回文中心,不参与到这种配对的计算中。
代码:
#include<bits/stdc++.h>
typedef long long LL;
using namespace std;
void solve() {
int n, k;
cin >> n >> k;
map<int, int> mapp;
for (int i = 0; i < n; i++) {
int t;
cin >> t;
mapp[t]++;
}
vector<int> a;
for (const auto &[x, y]: mapp) {
if (y & 1) {
a.push_back(x);
}
}
if ((int) a.size() <= 1) {
cout << "0" << endl;
return;
}
map<int, vector<int>> rec;
for (int i = 0; i < a.size(); ++i) {
int t = a[i];
rec[t % k].push_back(t);
}
int cnt = 0;
LL ans = 0;
for (auto &[x, cur]: rec) {
bool ok = true;
if (cur.size() & 1) {
cnt++;
ok = false;
if (cnt > 1) {
cout << "-1" << endl;
return;
}
}
sort(cur.begin(), cur.end());
if (ok) {
for (int i = 1; i < cur.size(); i += 2) {
ans += (cur[i] - cur[i - 1]) / k;
}
} else if (cur.size() > 1) {
int m = (int) cur.size();
vector<LL> pref(m);
vector<LL> suff(m);
pref[1] = cur[1] - cur[0];
for (int i = 3; i < m; i += 2) {
pref[i] = pref[i - 2] + cur[i] - cur[i - 1];
}
suff[m - 2] = cur[m - 1] - cur[m - 2];
for (int i = m - 4; i >= 0; i -= 2) {
suff[i] = suff[i + 2] + cur[i + 1] - cur[i];
}
LL minn = min(pref[m - 2], suff[1]);
for (int i = 2; i < m - 2; i += 2) {
minn = min(minn, pref[i - 1] + suff[i + 1]);
}
ans += minn / k;
}
}
cout << ans << endl;
}
int main() {
int t;
cin >> t;
while (t--)
solve();
return 0;
}
F.Non-academic Problem(联通分量)
题意:
给你一个连通的无向图,其顶点用 1 1 1到 n n n之间的整数编号。你的任务是尽量减少图中存在路径的顶点 1 ≤ u < v ≤ n 1\leq u\lt v\leq n 1≤u<v≤n对的数量。为了实现这一目标,你可以从图形中删除一条边。
请找出顶点对的最小数目!
分析:
整个图是连通的,删除边双内的边并不会影响连通性,删除桥边会变成两个连通块,一个大小为 x x x另一个则为 n − x n−x n−x。
对于一个大小为
x
x
x的连通块,编号最小的点有另外
x
−
1
x−1
x−1个点与其对应,以此类推,因此满足条件的点对数量是:
∑
i
=
1
x
−
1
i
=
x
(
x
−
1
)
2
\sum_{i=1}^{x-1}i=\frac{x(x-1)}{2}
i=1∑x−1i=2x(x−1)
考虑将图缩点为边双连通分量,然后只枚举桥边(即树边),对于所有情况取一个最小值即可。
代码:
#include<bits/stdc++.h>
typedef long long LL;
using namespace std;
const int N = 100010;
const int M = 200010;
int h[N], sz[N], dfn[N], low[N], id[N], szedcc[N], idx, n, m, timestamp, cntedcc;
stack<int> stk;
vector<int> G[N];
set<LL> S;
LL ans;
struct Edge {
int to, nxt;
} e[M];
void add(int a, int b) {
e[idx] = {b, h[a]};
h[a] = idx++;
}
void tarjan(int u, int from) {
stk.push(u);
dfn[u] = low[u] = ++timestamp;
for (int i = h[u]; i; i = e[i].nxt) {
int to = e[i].to;
if (!dfn[to]) {
tarjan(to, i);
low[u] = min(low[u], low[to]);
} else if (i != (from ^ 1))
low[u] = min(low[u], dfn[to]);
}
if (low[u] == dfn[u]) {
++cntedcc;
int y;
do {
y = stk.top();
stk.pop();
id[y] = cntedcc;
szedcc[cntedcc]++;
} while (y != u);
}
}
LL calc(LL x) {
return x * (x - 1) / 2;
}
void dfs(int u, int f) {
sz[u] = szedcc[u];
for (int to: G[u]) {
if (to == f)
continue;
dfs(to, u);
sz[u] += sz[to];
ans = min(ans, calc(sz[to]) + calc(n - sz[to]));
}
}
int main() {
int t;
cin >> t;
while (t--) {
cin >> n >> m;
idx = 2;
timestamp = cntedcc = 0;
S.clear();
for (int i = 0; i <= n; i++) {
h[i] = dfn[i] = szedcc[i] = sz[i] = id[i] = low[i] = 0;
vector<int>().swap(G[i]);
}
for (int i = 1, a, b; i <= m; i++) {
cin >> a >> b;
add(a, b);
add(b, a);
}
for (int i = 1; i <= n; i++) {
if (!dfn[i])
tarjan(i, -1);
}
for (int u = 1; u <= n; u++) {
for (int i = h[u]; i; i = e[i].nxt) {
int to = e[i].to;
int A = id[u], B = id[to];
if (A < B)
swap(A, B);
if (id[u] != id[to]) {
LL hash = 1ll * A * N + B;
if (S.count(hash))
continue;
S.insert(hash);
G[id[u]].emplace_back(id[to]);
G[id[to]].emplace_back(id[u]);
}
}
}
ans = calc(n);
dfs(1, 0);
cout << ans << endl;
}
return 0;
}
赛后交流
在比赛结束后,会在交流群中给出比赛题解,同学们可以在赛后查看题解进行补题。
群号: 704572101,赛后大家可以一起交流做题思路,分享做题技巧,欢迎大家的加入。