CodeTON Round 6 (Div. 2)

A. MEXanized Array

大致题意

构造一个数组,其长度为 n,最大值为 xMEXk,问这个数组的所有值的和最大是多少

思路

简单题,在 k>x+1n<k 的场景下无解(不可能构造一个 MEX 不可达的数组),然后就随便构造就行了,保证 MEX 之后,剩下所有值取最大就行

AC code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n, k, x;
cin >> n >> k >> x;
if (k > x + 1 || n < k) {
cout << -1 << endl;
continue;
}
int sum = 0;
for (int i = 0; i < k; ++i) sum += i;
for (int i = k; i < n; ++i) sum += (x == k ? x - 1 : x);
cout << sum << endl;
}
}
CPP

B. Friendly Arrays

大致题意

给出两个数组 ,允许选择任意次的 数组中的任意一个 ,然后让 ,问最终得到的数组 中所有的异或和最大和最小的可能

思路

某个比特为是 的情况下,在奇数个值异或和的结果则也是 ,而偶数个则为 。而或运算可以让 数组的每一个值的某些个位都变成 。基于此,只需要关心 的长度即可,若 为奇数,那么选尽可能多的 使得每个位都尽可能是 ,反之则尽可能不选,这样才能达到最大,同理可以得到最小的方案

AC code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n, m;
cin >> n >> m;
int sa = 0, sb = 0, tmp;
for (int i = 0; i < n; ++i) {
cin >> tmp;
sa ^= tmp;
}
for (int i = 0; i < m; ++i) {
cin >> tmp;
sb |= tmp;
}
cout << (n % 2 ? sa : sa & ~sb) << ' ' << (n % 2 ? sa | sb : sa) << endl;
}
}
CPP

C. Colorful Table

大致题意

有一个数组 ,长度为 ,然后有一个对应的矩阵 ,为 ,其每一个位置的值

问对于每个数字 ,在矩阵 ,中能够找到对应一个最小的矩形,此矩形包含了所有出现 的位置,求出这个矩形的大小

思路

对于任意一个值,假定其第一次在 中出现的位置为 ,它第一次出现在 地点一定是 ,同时其最后一次在矩阵中的位置一定是 ,其中 是在数组 ,中出现的,最后一个比当前值更大的下标

根据上面的规律,可以求出实际上每个值的位置,一定可以包裹比他大的那个值对应的矩阵,所以只需要根据值的大小排序一下他们在数组中第一次出现的位置,和最后一次出现的位置,然后从大到小遍历,保证小的值的区间能够覆盖到大的值的区间即可

AC code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n, k;
cin >> n >> k;
vector<bool> flag(k, false);
vector<pair<int, int>> data(k, {-1, -1});
for (int i = 0; i < n; ++i) {
int tmp;
cin >> tmp;
data[tmp - 1].first == -1 ? data[tmp - 1].first = data[tmp - 1].second = i : data[tmp - 1].second = i;
flag[tmp - 1] = true;
}
for (int i = k - 2; i >= 0; --i) {
data[i].first = !flag[i] ? data[i + 1].first : (data[i + 1].first != -1 ? min(data[i].first, data[i + 1].first) : data[i].first);
data[i].second = !flag[i] ? data[i + 1].second : (data[i + 1].first != -1 ? max(data[i].second, data[i + 1].second) : data[i].second);
}
for (int i = 0; i < k; ++i)
cout << (!flag[i] ? 0 : (data[i].second - data[i].first + 1) + (data[i].second - data[i].first + 1)) << ' ';

cout << endl;
}
}
CPP

D. Prefix Purchase

大致题意

有一个初始数组,每一个值都是 ,每次你可以选择花费 元,使得这个数组前 个元素加一,最多只能花费 元,问能够得到最大字典序的数组是什么

思路

首先需把 的值进行单调递增栈处理一下,毕竟价格相同或更低的同时 更大肯定有优势

回到题目中的字典序,意味着只有越前面的值越大即可,所以要尽可能满足最前面的值最大,所以直接把 丢给处理后的第一个值,看看最多第一个值可以到多少

处理完成第一个值后,那就意味着后面无论怎么贪心,第一个值一定要达到这个,否则肯定不如现在更好。另外,对于字典序而言,约前面的值价值越高,所以要尽可能让前面的值大,贪心一下即可

AC code

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
void solve() {
int _;
cin >> _;
for (int ts = 0; ts < _; ++ts) {
int n, k;
cin >> n;
vector<pair<int, int>> data;
for (int i = 0; i < n; ++i) {
int tmp;
cin >> tmp;
while (!data.empty() && data.back().first >= tmp) data.pop_back();
data.emplace_back(tmp, i);
}
data.emplace_back(INT_MAX, n - 1);
cin >> k;
vector<int> ans(data.size());
ans[0] = k / data.front().first;
k %= data.front().first;
for (int i = 1; i < data.size(); ++i) {
int diff = data[i].first - data[i - 1].first;
if (k < diff) {
break;
}
ans[i] = min(ans[i - 1], k / diff);
k -= diff * ans[i];
}
int cur = 0;
for (int i = 0; i < n; ++i) {
while (data[cur].second < i) cur++;
cout << ans[cur] << " \n"[i == n - 1];
}
}
}
CPP

CodeTON Round 6 (Div. 2)
https://blog.mauve.icu/2023/09/22/acm/codeforces/CodeTONRound6/
作者
Shiroha
发布于
2023年9月23日
许可协议