Less than 3
题目链接
Solution
首先我们可以发现一个性质,假设我们要变动第
i
i
i个位置上的字符,可以发现如果变动使得变动前后不存在连着相同的三个字符,那么一定有
i
−
1
i-1
i−1位置上的字符和
i
+
1
i+1
i+1位置上的字符不同。
我们在
0
0
0和
1
1
1之间划一条蓝分割线,
1
1
1和
0
0
0之间划一条红分割线,在变动过程中,我们可以发现,以下两个性质:
1
1
1、每次变动相当于左/右移一条红/蓝分割线。
2
2
2、任意时刻必须满足任意两条分割线之间的距离不能超过
2
2
2。
同时我们还可以发现,一条分割线可能会移出界。
到了这里,正解思路便很清晰了,枚举
s
/
t
s/t
s/t的
0
0
0位置前面有多少条分割线,之后便可以一一匹配分割线算代价,代价最小的便是答案。
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
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70#include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #define fo(i,j,l) for(int i=j;i<=l;++i) #define fd(i,j,l) for(int i=j;i>=l;--i) using namespace std; typedef long long ll; const ll N=152e2; int n,m; char s[N],t[N]; int a[N][2],b[N][2]; inline int abs(int o) {return (o<0)?(-o):o;} inline int min(int a,int b) {return a<b?a:b;} int main() { scanf("%d",&n); scanf("%s%s",s+1,t+1); int u=0,v=0; s[0]=t[0]='.'; fo(i,1,n)if(s[i]!=s[i-1])a[++u][1]=s[i]^48,a[u][0]=i; fo(i,1,n)if(t[i]!=t[i-1])b[++v][1]=t[i]^48,b[v][0]=i; int ans=n*n*2; fo(i,0,n){ int zd=max(i+u,v); int lj=0,ok=1; fo(l,1,zd){ int po=(l<=v)?b[l][0]:n+1; int co=(l<=v)?b[l][1]:2; int _po=(l<=i)?1:((l>i+u)?n+1:a[l-i][0]); int _co=(l<=i)?2:((l>i+u)?2:a[l-i][1]); if(_co!=co&&co!=2&&_co!=2)ok=0; lj=lj+abs(po-_po); } if(ok)ans=min(ans,lj); } u=v=0; fo(i,1,n)swap(s[i],t[i]); fo(i,1,n)if(s[i]!=s[i-1])a[++u][1]=s[i]^48,a[u][0]=i; fo(i,1,n)if(t[i]!=t[i-1])b[++v][1]=t[i]^48,b[v][0]=i; fo(i,0,n){ int zd=max(i+u,v); int lj=0,ok=1; fo(l,1,zd){ int po=(l<=v)?b[l][0]:n+1; int co=(l<=v)?b[l][1]:2; int _po=(l<=i)?1:((l>i+u)?n+1:a[l-i][0]); int _co=(l<=i)?2:((l>i+u)?2:a[l-i][1]); if(_co!=co&&co!=2&&_co!=2)ok=0; lj=lj+abs(po-_po); } if(ok)ans=min(ans,lj); } cout<<ans; }
最后
以上就是机灵奇异果最近收集整理的关于AtCoder agc 030 E的全部内容,更多相关AtCoder内容请搜索靠谱客的其他文章。
发表评论 取消回复