zoukankan      html  css  js  c++  java
  • CF264D

    题面
    官方题解

    题解概述:


    定义符号A~B表示序列A是序列B的子序列,A!~B反之。
    设操作序列为I,则有A~I,B!~I,C~I,D!~I。

    可得出条件①B!~C且D!~A,所以我们只要讨论满足这个条件的情况。

    分情况讨论:

    1. c1=c2,则可以进行操作c1,得到的状态仍满足条件①;
    2. c1!=c2,且B'!~C,此时可以进行操作c1;
    3. c1!=c2,且D'!~A,此时可以进行操作c2;
    4. c1!=c2,且D'~A,B'~C,此时无法进行操作,此情况无解;
      我们还发现第4种情况下,B=...xyxy,D=...yxyx,且B,D长度相等。
      另外,当B以xy结尾,D以yx结尾时,此情况一定无解。

    这就证明了当满足条件①时,无解当且仅当B以xy结尾,D以yx结尾。

    这样就可以对于数对(x,y)的每一个x,求出对应y的范围(一段区间),然后把其中以yx结尾的y减去(用前缀和实现),就可以得到每一个x的答案了。

    总复杂度O(n)。

    代码:

    #include<bits/stdc++.h>
    using namespace std;
    #define N 2000007
    #define ll long long
    int a[N],b[N],s[3][3][N];
    char ch[N];
    int tar(char c)
    {
    	if(c=='R')return 0;
    	if(c=='B')return 1;
    	return 2;
    }
    int main()
    {
    	//freopen("stone.in","r",stdin);
    	//freopen("stone.out","w",stdout);
    	int n,m,i,j,l,r;
    	ll ans=0;
    	scanf("%s",ch+1);
    	n=strlen(ch+1);
    	for(i=1;i<=n;i++)
    		a[i]=tar(ch[i]);
    	scanf("%s",ch+1);
    	m=strlen(ch+1);
    	for(i=1;i<=m;i++)
    		b[i]=tar(ch[i]);
    	for(int x=0;x<=2;x++)
    		for(int y=0;y<=2;y++)
    		{
    			for(i=2;i<=m;i++)
    				if(b[i-1]==x&&b[i]==y)
    					s[x][y][i]=1;
    			for(i=1;i<=m;i++)
    				s[x][y][i]+=s[x][y][i-1];
    		}
    	l=1,r=1;
    	while(r<m&&b[r]!=a[1])r++;
    	ans=r-l+1;
    	for(i=2;i<=n;i++)
    	{
    		if(l<=m&&a[i-1]==b[l])l++;
    		if(l>m)break;
    		if(r<m)
    		{
    			r++;
    			while(r<m&&b[r]!=a[i])r++;
    		}
    		if(l>r)continue;
    		int x=a[i-1],y=a[i];
    		ans+=r-l+1;
    		if(x!=y)ans-=s[y][x][r]-s[y][x][l-1];
    	}
    	printf("%lld
    ",ans);
    	return 0;
    }
    
  • 相关阅读:
    Rocket broker启动失败?
    jar下载慢,maven配置国内仓库
    Linux下搭建RocketMQ环境
    几种常见css布局
    原生JS快速实现拖放
    更换 grub 主题
    安装node环境以及cnpm
    用 hexo 快速搭建博客
    关于8080端口被占用问题的解决办法。
    idea的几个常用+实用的快捷键!
  • 原文地址:https://www.cnblogs.com/lishuyu2003/p/11502858.html
Copyright © 2011-2022 走看看