1 条题解

  • 0
    @ 2025-8-24 21:49:36

    自动搬运

    查看原文

    来自洛谷,原作者为

    avatar 文文殿下
    这家伙很勤奋,但还是什么都没有留下~~

    搬运于2025-08-24 21:49:36,当前版本为作者最后更新于2018-07-14 20:42:13,作者可能在搬运后再次修改,您可在原文处查看最新版

    自动搬运只会搬运当前题目点赞数最高的题解,您可前往洛谷题解查看更多

    以下是正文


    题意描述

    一句话描述:对于一个0/1序列,求出其中异或意义下回文的子串数量。

    题解

    我们可以看出,这个其实是一个对于异或意义下的回文子串数量的统计,什么是异或意义下呢?平常,我们对回文的定义是,对于任意ii,S[i]=S[ni+1]S[i]=S[n-i+1],而我们把相等改为异或操作,那么,当且仅当1100相匹配时,返回值为11 也就是 “真”。

    那么,我们可以尝试使用Manache算法来解决。当然,编程时,我们并不必真的去把0/1序列转换为数字序列,进行异或操作,这样会给自己增加一波常数(迷),我们构造一个to数组,to[x]to[x]数组的定义为 对于字符xx 我们允许匹配的对应字符,显然,to[0]=1to['0']='1',to[1]=0to['1']='0',特别的to[#]=# to['\#']='\#' to[$]=$to['\$']='\$' 。(此处'#'与'$'是Manache算法的分隔字符与防止溢出字符,可以自定义)。

    对于Manache算法有任何不了解的地方,可以戳!!!这里!!!,又看不懂的地方,也可以联系文文

    最后推一波自己博客(逃

    对于代码:

    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    const int maxn = 1000010;
    typedef unsigned long long ull;
    char SS1[maxn],S[maxn],to[500];
    int n,len[maxn],tot=1;
    int main() {
    	scanf("%d%s",&n,SS1+1);S[0]='$',S[1]='#';
    	for(register int i=1;i<=n;++i) S[++tot]=SS1[i],S[++tot]='#';
    	to['1']='0',to['0']='1',to['#']='#',to['$']='$';
    	int pos=1,mx=1;ull ans=0;
    	for(register int i=1;i<=tot;i+=2) {
    		len[i]=(i<mx?std::min(mx-i,len[(pos<<1)-i]):1);
    		while(S[i+len[i]]==to[S[i-len[i]]]) len[i]++;
    		if(len[i]+i>mx) {
    			mx=len[i]+i;pos=i;
    		}
    		ans+=len[i]>>1;
    	}
    	printf("%llu\n",ans);
    	return 0;
    }
    
    
    • 1

    信息

    ID
    2577
    时间
    1000ms
    内存
    125MiB
    难度
    5
    标签
    递交数
    0
    已通过
    0
    上传者