http://www.lydsy.com/JudgeOnline/problem.php?id=3160 (题目链接)

题意

  给定一个由'a'和'b'构成的字符串,求不连续回文子序列的个数。

Solution

  在膜拜了PoPoQQQ大爷的题解后,我觉得有必要自己写一发,感觉这道题倒还是可以理解的。

  不连续的回文子序列个数感觉并不是特别好求,而不连续的回文子序列个数=回文子序列个数-连续回文子序列个数。后者很好办,就是${Manacher}$板子,考虑没有任何限制的回文子序列个数怎么求。
  借用${Manacher}$的做法,先在字符串中添加分隔符,对于一个串${S}$长成这样:${aba}$,我们添加分隔符,那么得到${S'}$:${\$\#a\#b\#a\#}$

  我们令${f_i}$表示以${i}$为中心的对称字符对的个数,那么很显然,以${i}$为中心的回文子序列的个数就等于${2^{f_i}-1}$,那么整个答案就等于:$${\sum_{i=1}^{2*n+1} {2^{f_i}-1}}$$

  如果我们知道了${f_i}$,那么就可以算出答案了,而${f_i}$怎么求呢。

  这里假设${S}$和${S'}$的下标都从${0}$开始(因为多项式的次数是从${0}$开始的,这样的话就可以避免一些下标的细节问题),如果${S'}$中的两个字母比如说这两个${a}$,它们是关于中间的${b}$对称的,想一想这两个${a}$与${b}$位置之间的关系:${S'_2=a,S'_6=a,S'_4=b}$,可以得到它们的下标:${x=2,y=6,mid=4}$,很显然:${x+y=2*mid}$。因为${S'}$中字母的下标都是偶数,我们将两边同时除以${2}$,得到这样一个式子:${x/2+y/2=mid}$。那么${x/2}$和${y/2}$表示什么呢,是不是就是${S'}$中的字母在原串${S}$中的位置${+1}$。那么我们可以得到这样一个结论:如果原串中两个字母${S_x=S_y}$,它们就会关于${S'_{(x+1)+(y+1)}}$对称。

  那么我们就可以写出${f_i}$的表达式了:$${f_i=\lfloor{  \frac{1+\sum_{j=0}^{i-2} {[s_j=s_{i-2-j}]}}  {2}  }\rfloor }$$

  这是一个卷积的形式,我们很容易想到用${FFT}$来解决这个东西。首先考虑${a}$对答案的贡献,令${a=1,b=0}$,求一遍${FFT}$;再考虑${b}$对答案的贡献,令${a=0,b=1}$,再求一遍${FFT}$;两者相加再加${1}$最后除以${2}$取下整就是${f}$了。代入之前的结论中即可出解。

  话说我还是不会构造多项式,还是写一写吧。。比如说样例:${aba}$。我们考虑算${a}$的贡献:

$${A(x)=1*x^0+0*x^1+1*x^2}$$

$${B(x)=1*x^0+0*x^1+1*x^2}$$

  那么最后的${f_i}$就是${A(x)*B(x)}$的第${i-1}$项的系数(次数为${i-2}$)。

细节

  细节还是蛮多的,搞清楚下标始终是个问题。。注意数组大小,以及计算${f}$的时候要开LL。

代码

// bzoj3160
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<complex>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define MOD 1000000007
#define inf 1ll<<60
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; typedef complex<double> E;
const int maxn=400010;
int bin[maxn],rev[maxn],f[maxn],p[maxn],n,L;
E a[maxn],b[maxn];
char s[maxn],st[maxn]; void FFT(E *a,int f) {
for (int i=0;i<n;i++) if (i<rev[i]) swap(a[i],a[rev[i]]);
for (int i=1;i<n;i<<=1) {
E wn(cos(Pi/i),f*sin(Pi/i));
for (int p=i<<1,j=0;j<n;j+=p) {
E w(1,0);
for (int k=0;k<i;k++,w*=wn) {
E x=a[k+j],y=a[k+j+i]*w;
a[k+j]=x+y;a[k+j+i]=x-y;
}
}
}
}
int Manacher(char *r,int len) {
st[0]='$';st[1]='#';
for (int i=0;i<len;i++) {
st[(i+1)<<1]=r[i];
st[(i+1)<<1|1]='#';
}
int mx=0,id,tot=0;
for (int i=1;i<2*len+1;i++) {
if (i<mx) p[i]=min(mx-i,p[id*2-i]);
while (st[i+p[i]]==st[i-p[i]]) p[i]++;
if (p[i]+i>mx) mx=p[i]+i,id=i;
tot=(tot+p[i]/2)%MOD;
}
return tot;
} int main() {
scanf("%s",s);
int len=strlen(s);
bin[0]=1;for (int i=1;i<maxn;i++) bin[i]=(bin[i-1]<<1)%MOD;
for (n=1;n<=len<<1;n<<=1) L++;
for (int i=0;i<n;i++) rev[i]=(rev[i>>1]>>1) | ((i&1)<<(L-1));
for (int i=0;i<len;i++) {
if (s[i]=='a') a[i]=1;
else a[i]=0;
}
FFT(a,1);
for (int i=0;i<n;i++) b[i]=a[i]*a[i];
memset(a,0,sizeof(a));
for (int i=0;i<len;i++) {
if (s[i]=='b') a[i]=1;
else a[i]=0;
}
FFT(a,1);
for (int i=0;i<n;i++) b[i]+=a[i]*a[i];
FFT(b,-1);
LL ans=0;
for (int i=2;i<2*len+1;i++) f[i]+=(LL)(b[i-2].real()+0.5)/n;
for (int i=2;i<2*len+1;i++) ans=(ans+bin[(f[i]+1)>>1]-1)%MOD;
printf("%lld",(ans+MOD-Manacher(s,len))%MOD);
return 0;
}

【bzoj3160】 万径人踪灭的更多相关文章

  1. [bzoj3160]万径人踪灭_FFT_Manacher

    万径人踪灭 bzoj-3160 题目大意:给定一个ab串.求所有的子序列满足:位置和字符都关于某条对称轴对称而且不连续. 注释:$1\le n\le 10^5$. 想法: 看了大爷的题解,OrzOrz ...

  2. BZOJ3160 万径人踪灭 字符串 多项式 Manachar FFT

    原文链接http://www.cnblogs.com/zhouzhendong/p/8810140.html 题目传送门 - BZOJ3160 题意 给你一个只含$a,b$的字符串,让你选择一个子序列 ...

  3. BZOJ3160 万径人踪灭(FFT+manacher)

    容易想到先统计回文串数量,这样就去掉了不连续的限制,变为统计回文序列数量. 显然以某个位置为对称轴的回文序列数量就是2其两边(包括自身)对称相等的位置数量-1.对称有啥性质?位置和相等.这不就是卷积嘛 ...

  4. BZOJ3160万径人踪灭

    Description Input & Output & Sample Input & Sample Output HINT 题解: 题意即求不连续但间隔长度对称的回文串个数. ...

  5. BZOJ3160: 万径人踪灭

    设a[i]=bool(s[i]=='a'),b[i]=bool(s[i]=='b'),考虑a和a.b和b的卷积,由于卷积是对称的,就可以统计出不连续回文子串个数了.可能说得比较简略.再用manache ...

  6. bzoj千题计划302:bzoj3160: 万径人踪灭

    https://www.lydsy.com/JudgeOnline/problem.php?id=3160 不连续的回文串数量=所有的回文序列数量-连续的回文子串 连续的回文子串: manacher ...

  7. BZOJ3160:万径人踪灭(FFT,Manacher)

    Solution $ans=$回文子序列$-$回文子串的数目. 后者可以用$manacher$直接求. 前者设$f[i]$表示以$i$为中心的对称的字母对数. 那么回文子序列的数量也就是$\sum_{ ...

  8. BZOJ3160 万径人踪灭 【fft + manacher】

    题解 此题略神QAQ orz po神牛 由题我们知道我们要求出: 回文子序列数 - 连续回文子串数 我们记为ans1和ans2 ans2可以用马拉车轻松解出,这里就不赘述了 问题是ans1 我们设\( ...

  9. BZOJ3160: 万径人踪灭(FFT,回文自动机)

    BZOJ传送门: 解题思路: FFT在处理卷积时可以将自己与自己卷,在某一种字母上标1其他标0,做字符集次就好了. (回文就是直接对称可以联系偶函数定义理解,根据这个性质就可以将字符串反向实现字符串匹 ...

  10. 多项式 之 快速傅里叶变换(FFT)/数论变换(NTT)/常用套路【入门】

    原文链接https://www.cnblogs.com/zhouzhendong/p/Fast-Fourier-Transform.html 多项式 之 快速傅里叶变换(FFT)/数论变换(NTT)/ ...

随机推荐

  1. 6.Xilinx RapidIO核仿真与包时序分析

    转自https://www.cnblogs.com/liujinggang/p/10123498.html 一.软件平台与硬件平台 软件平台: 操作系统:Windows 8.1 64-bit 开发套件 ...

  2. Java 多线程(五)之 synchronized 的使用

    目录 1 线程安全 2 互斥锁 3 内置锁 synchronized 3.1 普通同步方法,锁是当前实例对象(this) 3.1.1 验证普通方法中的锁的对象是同一个. 3.1.2 验证不同的对象普通 ...

  3. Spring的单例模式底层实现学习笔记

    单例模式也属于创建型模式,所谓单例,顾名思义,所指的就是单个实例,也就是说要保证一个类仅有一个实例.单例模式有以下的特点:①单例类只能有一个实例②单例类必须自己创建自己的唯一实例③单例类必须给所有其他 ...

  4. MVC的多页面后台管理系统

    MVC的多页面后台管理系统 同样功能的后台管理系统,也是可以使用 ASP.NET MVC .Web API 和JQuery 来制作. 所有的功能都与Angular js的单页面相同.应用层所有的方法都 ...

  5. CSS 背景实例

    CSS 背景属性属性 描述background 简写属性,作用是将背景属性设置在一个声明中.background-attachment 背景图像是否固定或者随着页面的其余部分滚动.background ...

  6. VirtualBox虚拟机怎么导入已经存在的vdi文件

    VirtualBox虚拟机怎么导入已经存在的vdi文件 第一章 1.原因 早上一不小心将virtualBox 卸载了,(不知道怎么了, 里面得虚拟机全部都没有了,但是vdi文件还在) 2.解决办法 直 ...

  7. Redis源码阅读(二)高可用设计——复制

    Redis源码阅读(二)高可用设计-复制 复制的概念:Redis的复制简单理解就是一个Redis服务器从另一台Redis服务器复制所有的Redis数据库数据,能保持两台Redis服务器的数据库数据一致 ...

  8. PBFT_拜占庭容错算法

    根据论文<Practical Byzantine Fault Tolerance and Proactive Recovery>整理 Practical byzantine fault t ...

  9. android listview addheaderview viewpager

    just set viewPager's onTouchListener,like this: viewPager.setOnTouchListener(new OnTouchListener() { ...

  10. babel的使用(关于使用async报错的问题)

    一.配置文件.babelrc .babelrc 文件存放在项目的根目录下. { "presets": [], "plugins": [] } presets 字 ...