Portal --> pkuscD2T2(loj6436)

Solution

  个人觉得是道很好的法法塔题qwq

  一开始的时候想偏了想到了另一种法法塔处理字符串匹配之类的奇怪技巧(万径人踪灭qwq了解一下),然后非常天真的想把问号拿出来单独处理一波然后越走越远。。。

  好吧正解其实是(感觉貌似还是想的有点绕?应该有更加简便的思维方式不过qwq不管了qwq)

  因为有一个问号所以其实挺难直接得到哪些长度能够匹配,那不妨换个思路把所有肯定不能匹配的长度去掉那剩下的就是答案了

  肯定不能匹配的长度显然就是\(0\)和\(1\)对上了,假设我们现在知道位置\(x\)有一个\(0\),位置\(y\)有一个\(1\),那么考虑当匹配长度\(k\)满足什么条件的时候这两个位置是对上的,那么满足这样条件的\(k\)显然就是可以去掉的了,那么可以得到(假设字符串从\(0\)开始,\(x<y\) ,记\(y-x=len\),字符串总长度为\(n\)):

\[x=y-(n-k)
\]

  化一下就是

\[k=n-len
\]

​  除此之外,这个匹配的话其实就是\(kmp\)那一套的感觉,因此用类似\(kmp\)推\(next\)数组那样的想法(额其实也不是吧不知道怎么说就是各种推来推去有点像的感觉qwq)去推一下这个串会发现这样一个性质:记两个匹配串的开头位置的差为\(K\),如果存在一对下标差为\(len\)的\(01\),那么满足\(K|len\)的长度都是无法匹配的

  具体一点就是这样

  第一个蓝点和红点的下标差是\(2K\)但是会影响到开头位置差为\(K\)的匹配

  所以得知了\(01\)对的下标差之后我们再枚举一下倍数筛一下就可以了

  

  至于\(01\)对的下标差,我们把\(x-y\)转成\(x+(n-1-y)\),这样的话就可以用法法塔来做了(注意\(s\)是从\(0\)开始的),考虑这样两个函数

\[\begin{aligned}
g(x)&=\sum [s[n-1-i]==0]x^i\\
f(x)&=\sum [s[i]==1]x^i\\
\end{aligned}
\]

  我们把指数看成上面的\(x\)和\(n-1-y\)然后直接这两个多项式乘一下,结果为\(h(x)\)

  那么对于\(h(x)\)中指数小于\(n\)的部分,表示的是\(x>y\)的情况(因为这部分的指数\(-(n-1)\)得到的是负数),取反一下就可以得到下标差了,大于等于\(n\)的部分直接减就好了

  最后枚举倍数筛一波然后输出就好了,注意这里的是首位的下标差所以\(i\)对应的长度应该是\(n-i\)

​   

  代码大概长这个样子(然而貌似用fft会更快但我也不知道为啥就写了ntt不管了qwq)

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
using namespace std;
const int MAXN=5*(1e5)+10,MOD=998244353,G=3;
int ksm(int x,int y);
namespace NTT{/*{{{*/
int A[MAXN*4],B[MAXN*4],W[MAXN*4][2],rev[MAXN*4];
int invg,len,invlen;
void init(){
invg=ksm(G,MOD-2);
for (int i=0;i<=20;++i){
W[1<<i][0]=ksm(G,(MOD-1)/(1<<i));
W[1<<i][1]=ksm(invg,(MOD-1)/(1<<i));
}
}
void prework(int *a,int *b,int n,int m){
for (int i=0;i<len;++i) A[i]=B[i]=0;
int bit=0;
for (len=1;len<n+m;len<<=1,++bit);
rev[0]=0;
for (int i=1;i<len;++i) rev[i]=(rev[i>>1]>>1)|((i&1)<<(bit-1));
for (int i=0;i<=n;++i) A[i]=a[i];
for (int i=0;i<=m;++i) B[i]=b[i];
invlen=ksm(len,MOD-2);
}
void ntt(int *a,int op){
int w,w_n,t,u;
for (int i=0;i<len;++i)
if (rev[i]>i) swap(a[i],a[rev[i]]);
for (int step=2;step<=len;step<<=1){
w_n=op==1?W[step][0]:W[step][1];
for (int st=0;st<len;st+=step){
w=1;
for (int i=0;i<(step>>1);++i){
t=1LL*a[st+i+(step>>1)]*w%MOD;
u=a[st+i];
a[st+i]=(1LL*u+t)%MOD;
a[st+i+(step>>1)]=(1LL*u-t+MOD)%MOD;
w=1LL*w*w_n%MOD;
}
}
}
if (op==1) return;
for (int i=0;i<len;++i) a[i]=1LL*a[i]*invlen%MOD;
}
void Ntt(int *a,int *b,int n,int m){
prework(a,b,n,m);
ntt(A,1);
ntt(B,1);
for (int i=0;i<len;++i) A[i]=1LL*A[i]*B[i]%MOD;
ntt(A,-1);
}
void debug(int *a,int n){
prework(a,a,n,n);
ntt(a,1);
ntt(a,-1);
}
}/*}}}*/
char s[MAXN];
int ori1[MAXN],ori0[MAXN],res[MAXN*2];
int out[MAXN];
ll ans;
int n,m,lens;
void prework(); int main(){
#ifndef ONLINE_JUDGE
freopen("a.in","r",stdin);
#endif
scanf("%s",s);
n=strlen(s);
NTT::init();
prework();
NTT::Ntt(ori1,ori0,n,n);
for (int i=0;i<=n*2;++i) res[i]=NTT::A[i];
for (int i=0;i<n;++i) out[n-1-i]|=res[i];
for (int i=n;i<2*n-1;++i) out[i-(n-1)]|=res[i];
for (int i=1;i<=n;++i)
for (int j=i;j<=n&&!out[i];j+=i)
if (out[j]) out[i]=1;
ans=0;
for (int i=1;i<=n;++i)
if (!out[n-i]) ans^=1LL*i*i;
printf("%lld\n",ans);
} int ksm(int x,int y){
int ret=1,base=x;
for (;y;y>>=1,base=1LL*base*base%MOD)
if (y&1) ret=1LL*ret*base%MOD;
return ret;
} void prework(){
for (int i=0;i<n;++i)
ori1[i]=s[i]=='1',ori0[i]=s[i]=='0';
reverse(ori0,ori0+n);
}

【loj6436】【pkusc2018】神仙的游戏的更多相关文章

  1. LOJ6436 [PKUSC2018] 神仙的游戏 【FFT】

    题目分析: 题目要求前后缀相同,把串反过来之后是一个很明显的卷积的形式.这样我们可以完成初步判断(即可以知道哪些必然不行). 然后考虑一下虽然卷积结果成立,但是存在问号冲突的情况. 箭头之间应当不存在 ...

  2. [LOJ6436][PKUSC2018]神仙的游戏

    loj description 给你一个只有01和?的字符串,问你是否存在一种把?改成01的方案使串存在一个长度为\(1-n\)的\(border\).\(n\le5\times10^5\) sol ...

  3. BZOJ5372: [Pkusc2018]神仙的游戏

    BZOJ5372: [Pkusc2018]神仙的游戏 https://lydsy.com/JudgeOnline/problem.php?id=5372 分析: 如果\(len\)为\(border\ ...

  4. BZOJ5372: PKUSC2018神仙的游戏

    传送门 Sol 自己还是太 \(naive\) 了,上来就构造多项式和通配符直接匹配,然后遇到 \(border\) 相交的时候就 \(gg\) 了 神仙的游戏蒟蒻还是玩不来 一个小小的性质: 存在长 ...

  5. bzoj 5372: [Pkusc2018]神仙的游戏

    Description 小D和小H是两位神仙.他们经常在一起玩神仙才会玩的一些游戏,比如"口算一个4位数是不是完全平方数". 今天他们发现了一种新的游戏:首先称s长度为len的前缀 ...

  6. loj 6436 PKUSC2018 神仙的游戏

    传送门 好妙蛙 即串\(s\)长度为\(n\)首先考虑如果一个长度为\(len\)的\(border\)存在,当且仅当对所有\(i\in[1,len],s[i]=s[n-len+i]\),也就是所有模 ...

  7. BZOJ5372 PKUSC2018神仙的游戏(NTT)

    首先有一个想法,翻转串后直接卷积看有没有0匹配上1.但这是必要而不充分的因为在原串和翻转串中?不能同时取两个值. 先有一些结论: 如果s中长度为len的前缀是border,那么其存在|s|-len的循 ...

  8. [PKUSC2018]神仙的游戏(FFT)

    给定一个01?串,对所有len询问是否存在一种填法使存在长度为len的border. 首先有个套路的性质:对于一个长度为len的border,这个字符串一定有长度为n-len的循环节(最后可以不完整) ...

  9. [PKUSC2018]神仙的游戏

    题目 画一画就会发现一些奇诡的性质 首先如果\(len\)为一个\(\operatorname{border}\),那么必然对于\(\forall i\in [1,len]\),都会有\(s_i=s_ ...

  10. 【LOJ6436】【PKUSC2018】神仙的游戏(NTT)

    [LOJ6436][PKUSC2018]神仙的游戏(NTT) 题面 LOJ 题解 看到\(zsy\)从\(PKUSC\)回来就秒掉了这种神仙题 吓得我也赶快看了看\(PKUSC\)都有些什么神仙题 然 ...

随机推荐

  1. webpack Error: Cannot find module 'webpack/lib/Chunk' Extract-text-webpack-plugin 分离CSS

    深入浅出webpack 1.5章节使用Extract-text-webpack-plugin分离css 安装插件后打包提示错误 Error: Cannot find module 'webpack/l ...

  2. 《图解 HTTP 》阅读 —— 第四章

    第4章 返回结果的HTTP状态码 1XX 接收的请求正在处理 2XX 请求被处理 200 请求成功 204 请求成功,但是没有返回数据 206 客户端进行了范围请求 3XX 重定向 301 永久性重定 ...

  3. 算法笔记(c++)--回文

    算法笔记(c++)--回文 #include<iostream> #include<algorithm> #include<vector> using namesp ...

  4. Leetcode_2. Add_Two_Number

    2. Add_Two_Number 用两个非空链表分别表示两个非负整数,链表的节点表示数字的位,链表头表示数字的低位,链表尾表示数字高位.求两个链表所表示数字的和. 比如: Input: (2 -&g ...

  5. SVN服务器搭建及客户端配置

    为什么要使用SVN? 在程序的编写过程中,每个程序员都会负责开发一个或多个模块,且开发中会生成很多不同的版本, 这就需要程序员有效的管理代码,在需要的时候可以迅速,准确取出相应的版本. Subvers ...

  6. 团队计划会议(二)——WBS

    一.会议及WBS 因为是第一次开发android应用,所以我们对这次开发心里也没底,最后我们商量暂时先实现主要的几个骨架功能,之后再慢慢完善. 会议期间,我们根据自己的能力大致先估算了完成这些功能需要 ...

  7. Floyd模板(详细操作最基础版)

    #include<cstdio> #include<iostream> using namespace std; #define MAX 500 #define INFE 1& ...

  8. <<世界是数字的>>读书笔记

    <世界是数字的>这本书是大学职业规划老师介绍个我读的,从着本中我学到了很多. 第一章,计算机里有什么.这个问题可以从两方面来看:逻辑上或者说功能上的组成,即每一部分是什么.做什么.怎样做. ...

  9. vs调试iisExpress经常卡死

    最近调试一个项目时,电脑经常卡死,不得不强制重启,一直不知道iisExpress为何卡死的. 想了很多办法,强制删除bin里面的文件,结果不行: 企图删除iisExpress虚拟目录中的文件也不行: ...

  10. FTP渗透测试

    在这篇文章中,我们将学习如何在CentOS机器配置ProFTPD的服务.之后,我们将进行渗透测试,以评估FTP服务的安全性,那么我们也将学习漏洞的对策. 在CentOS Linux机器的安装和配置FT ...