目录

2020/07/25–2020/07/31周报(week12)

本周推荐

by iuiou

by QuantumBolt

这个题是博弈论的题: 分析一下: 设$p_i$表示$n$个数二进制位第$i$位为$1$的个数 若$2|p_i$那么先手和后手对$i$位的结果没有影响 从而我们需要找到$j$,使得$2|p_j=1$这样才能导致先后手的结果在第$j$位不同 现在我们的任务就是找打最大的$j$位,使结果最大化

1. 若$\frac{P_j-1}{2}$是偶数,那么先手的就一定赢 先手在该位先取一个1,然后跟着后手选,那么后手会选择偶数个1,结果为:先手在该位值为1,后手为0。

2. 若$\frac{P_j-1}{2}$是奇数,分两种情况

(1). 若$2|n = 1$先手就必输: 先手在该位先取一个1,然后跟着后手选,那么后手会择偶数个1,结果为:先手在该位值为1,后手为0。

(2). 若$2|n$那么先手必赢 先手第一步选一个0,然后将状态转为$2|n=1$的局面,且轮到后手先选择,上面已经证明,这种局面先选择的必输。

团队训练

2020.7.25 牛客多校第六场

2020.7.27 牛客多校第七场

范泽恒

专题

比赛

题目

恭天祥

专题

比赛

题目

刘怀远

专题

比赛

题目

为什么这破题当初能卡那么久

ac代码

ac代码

#include<bits/stdc++.h>
using namespace std;
 
typedef long long ll;
const int maxn = 2e5 + 5;
const double pi = acos(-1);
const int mod = 998244353;
 
int t, n;
char s[131074];
 
int cal(int l, int r, int c){
	int cnt = 0;
	for(int i = l; i <= r; ++i) if(s[i] != c) ++cnt;
	return cnt;
}
 
int dfs(int l, int r, char c){
	if(l == r) return s[l] != c;
	int res = 3e7;
	int mid = (l + r) >> 1;
	res = min(res, dfs(l, mid, c + 1) + cal(mid + 1, r, c));
	res = min(res, dfs(mid + 1, r, c + 1) + cal(l, mid, c));
	return res;
}
 
int main(){
	scanf("%d", &t);
	while(t--){
		scanf("%d\n", &n);
		scanf("%s", s + 1);
		printf("%d\n", dfs(1, n, 'a'));
	}
}