这些题都很nice,抓住关键点就能突破。
1、Maximum Subsequence Value
从\(n\)个数中挑\(k\)个数,使得\(\sum2^i\)最大(将这\(k\)个数用二进制表示,如果二进制第 \(i\) 位为1的数的个数多于\(max(1,k-2)\)个,就加上\(2^i\))。
题解
性质:选高位为1的数比选多个低位为1的数更优,\(1 + 2^1 + 2^2+···+2^{n-1} <2^n\)
当\(n>=3\)时,\(k =3\)能取得最大的\(\sum2^i\)。也就是说,当\(k>3\)时,多选的\(k-3\)个数不能使得原本第\(i\)位的0(\(k =3\)时)变为有效的1,既第\(i\)位全为1,也只有\(k-3\)个数,小于\(max(1,k-2)\)。
int main()
{
int n;
cin >> n;
long long a[505];
myfor(i, 1, n) cin >> a[i];
if (n <= 3) {
long long ans = a[1];
myfor(i, 1, n) ans |= a[i];
cout << ans << endl;
}
else {
long long ans = 0;
myfor(i, 1, n) myfor(j, i + 1, n) myfor(k, j + 1, n) ans = max(ans, a[i]|a[j]|a[k]);
cout << ans << endl;
}
return 0;
}
2、Solve The Maze
给一张二维迷宫,出口是\((n,m)\)
- . 表示这个位置可以行走
- #表示这个位置是墙
- G表示这个位置上有一个好人
- B表示这个位置上有一个坏人
问是否存在一种方案:在某些位置上放墙,既令\(map[i][j] = '\#'\),使得好人走出迷宫而坏人不能走出迷宫。
题解
在坏人的周围都放上墙,然后从出口出发,看是否能走到所有好人所在的位置。
3、Tree Shuffling(好题)
一棵树有\(n\)个节点,每个节点有三个值:\(a\)(花费),\(b\)(已有的值),\(c\)(想要变成的值)
操作:在任意一颗子树中选择\(k\)个节点,然后按照你的意愿排列这\(k\)个节点
操作代价:假设这颗子树的根是\(u\),\(operation\_cost = 2 * k * a[u]\)
问任意次操作后,将所有的节点的\(b\)变成\(c\)的最小代价是多少?
\(b,c\in \{0,1\}\)
题解
贪心,从花费较小的节点开始操作起
思路比较简单,就是不会写
pair<ll, ll> DFS(int u, int pa, int cost) {
pair<int, int> now; // 统计这颗子树需要对几个节点进行重新排列
if (b[u] != c[u]) {
now.first += b[u];
now.second += c[u];
}
for (int i = head[u]; i != -1; i = e[i].next) {
int v = e[i].to;
if (v == pa) continue;
pair<ll, ll> nex = DFS(v, u, min(cost, a[v]));
now.first += nex.first;
now.second += nex.second;
}
int cnt = min(now.first, now.second);
ans += 2ll * cnt * cost; // cost: root -> u 这条链上的最小花费
now.first -= cnt;
now.second -= cnt;
return now;
}
4、Odd-Even Subsequence
给一个序列\(a\),从中挑选\(k\)个数构成序列\(s\)(不改变数的位置)。定义\(x = min(max\{s_1,s_3,s_5,s_7,···\},max\{s_2,s_4,s_6,s_8,···\})\),问最小的\(x\)是多少?
题解
最小化最大值,考虑二分枚举答案,检查以下两种情况即可,换言之,我们需要构造出这样一个子序列\(s\)
- 答案在奇序列中
- 答案在偶序列中
我的想法是:假设每次枚举的答案\(mid\)一定在\(a\)中(用\(mid\)构造子序列\(s\)),然后发现\(check\)的时候,非常不好做,因为必须判断假设是否成立。
/*
不断缩小答案的上界
*/
bool check(int x) {
// 答案在偶序列
for (int i = 1, cnt = 0; i <= n; i++) {
if (((cnt & 1) && (a[i] <= x)) || (cnt % 2 == 0)) cnt++; // 构造合适的s(贪心)
if (cnt >= k) return true;
}
// 答案在奇序列
for (int i = 1, cnt = 0; i <= n; i++) {
if (((cnt % 2 == 0) && (a[i] <= x)) || (cnt & 1)) cnt++;
if (cnt >= k) return true;
}
return false;
}
5、AND, OR and square sum
给一个序列\(a\),有如下操作:
- 任选两个数\(a_i\),\(a_j\),令\(a_i^{'}= a_i \& a_j\),\(a_j^{'} = a_i |a_j\)
可以操作任意次,令\(ans = \sum_{i = 1}^na_i^{'2}\),最大化\(ans\)。
题解
假设\(x = a_i |a_j\),有\(x >= a_i\)且\(x >= a_j\),所以我们可以让\(a\)中的最大值不断变大,而\(ans\)不会变小。按照这个思路有:令\(a_1 = a_{max}\)(按降序排序),有\(a_1 = a_1 | a_2\),\(a_2 = a_1 \& a_2\)、\(a_1 = a_1 | a_3\),\(a_3 = a_1 \& a_3\)、······、\(a_1 = a_1 | a_n\),\(a_n = a_1 \& a_n\)。用相同的办法求出\(a_2^{'}\),\(a_3^{'}\),...,\(a_n^{'}\),既能求得最大的\(ans\)。
显然,这样计算需要两次循环,复杂度\(O(n^2)\)。实际上,给出的操作有个性质:不会改变每位包含1的个数,既\(a_i^{'} = a_i \& a_j\),\(a_j^{'} = a_i | a_j\) 把 \(a_i\) 的二进制位上的 1 给了 \(a_j\),但总的1的个数并没有改变。
所以,我们只需要统计二进制每位有多少个1,然后组成一个个数就行了。
妙啊,做\(cf\)的题始终差那临门一脚!
int main()
{
int x;
cin >> n;
myfor(i, 1, n) {
cin >> x;
int t = 0;
while(x) {
cnt[t++] += x & 1;
x >>= 1;
}
}
ll ans = 0;
myfor(i, 1, n) {
ll x = 0;
myfor(j, 0, 20) if (cnt[j]) {
x += (1 << j);
cnt[j]--;
}
ans += x * x;
}
cout << ans << endl;
return 0;
}