• agc030_e Less than 3


    agc030_e Less than 3

    https://atcoder.jp/contests/agc030/tasks/agc030_e

    NXlVZd.png

    Tutorial

    https://img.atcoder.jp/agc030/editorial.pdf

    观察一次操作进行的条件.

    假如要对位置(x)进行操作,那么(x-1,x+1)的字符就必须不同,否则在操作前或后是不合法的.

    所以除非在边界操作,01的联通块顺序和数量是不会改变的.

    我们在01之间插入红色隔板,在10之间插入蓝色隔板,得到

    NXlqYt.png

    对于边界的情况,我们可以假设左右边界外分别有无限多隔板,且满足红蓝交替出现

    NX1dAA.png

    由此我们可以将问题转化为

    • 有若干个隔板,除了两个在边界的隔板之外,任何隔板之间的距离只能为(1)(2).且红蓝交替出现.
    • 一次操作可以将某个隔板向前或向后移动(1),且不改变相对位置,且移动后仍满足上述条件

    我们可以对于(s,t)之间的隔板建立对应关系,则答案下界为对应隔板之间的距离和.

    考虑这个下界是否总是可以达到,我们将隔板按照与其对应隔板的位置关系分为"向左,向右,不动"3类.由于隔板之间的距离为(1,2)且顺序不改变,所以没有向右和向左的隔板相邻,所以一定是形如"向右,向右,不动,向左,向左,不动"这样的顺序.观察发现一个向左或向右连续段里面总有一个隔板可以向目标位置移动.

    所以我们枚举对应关系即可,最多有(1)个左边界的隔板与右边界的隔板对应,所以一共有(O(n))种对应关系,时间复杂度(O(n^2))

    Code

    #include <cmath>
    #include <cstdio>
    #include <iostream>
    #include <vector>
    #define debug(...) fprintf(stderr,__VA_ARGS__)
    #define fi first
    #define se second
    using namespace std;
    template<class T> void rd(T &x) {
    	x=0; int f=1,ch=getchar();
    	while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
    	while(ch>='0'&&ch<='9'){x=x*10-'0'+ch;ch=getchar();}
    	x*=f;
    }
    template<class T> inline bool Cmin(T &x,T y) {return x>y?x=y,1:0;}
    const int inf=1e9;
    const int maxn=5000+5;
    int n; char s[maxn],t[maxn];
    vector< pair<int,int> > S,T;
    int sol(int a,int b) {
    	int an=0;
    	while(S[a].fi!=n||T[b].fi!=n) an+=abs(S[a++].fi-T[b++].fi);
    	return an;
    }
    void init(char *s,vector< pair<int,int> > &S) {
    	for(int i=0;i<n;++i) S.push_back(make_pair(0,i&1));
    	if((n&1)==(s[1]=='1')) S.push_back(make_pair(0,n&1));
    	for(int i=1;i<n;++i) if(s[i]!=s[i+1]) {
    		int c=S.back().se^1; 
    		S.push_back(make_pair(i,c));
    	}
    	for(int i=0;i<2*n;++i) {
    		int c=S.back().se^1;
    		S.push_back(make_pair(n,c));
    	}
    }
    int main() {
    	rd(n);
    	scanf("%s",s+1);
    	scanf("%s",t+1);
    	init(s,S),init(t,T);
    	int an=inf;
    	for(int i=0;i<=n;++i) if(S[i].se==T[n].se) Cmin(an,sol(i,n));
    	for(int i=0;i<=n;++i) if(S[n].se==T[i].se) Cmin(an,sol(n,i));
    	printf("%d
    ",an);
    	return 0;
    }
    
  • 相关阅读:
    .NET高级代码审计(第13课)反序列化Gadget之详解ObjectDataProvider完结篇
    sh脚本语法 1
    英语词汇替换词
    关于创业感悟第一篇关于夫妻店
    2021年度总结暨2022年归来
    测试
    Windows10专业版安全中心页面提示不可用 Windows defender 页面不可用 完美解决办法 在这里
    jmeter配置mysql插件以及连接
    centos更新安全补丁包
    centos配置yum源阿里
  • 原文地址:https://www.cnblogs.com/ljzalc1022/p/13230835.html
Copyright © 2020-2023  润新知