正题

题目链接:https://www.luogu.com.cn/problem/P4199


题目大意

给出一个只包含\(a,b\)的字符串

求有多少个不连续的回文子序列(字母回文,位置对称)

\(1\leq n\leq 10^5\)


解题思路

这个不连续一看就很nt,考虑求出所有的再减去连续的

对于每个对称轴来说,\(a,b\)分开考虑贡献再乘起来再减去一(两个都为空)。

加入计算\(a\)的贡献,就是看有多少对\(a\)以它为对称轴,设为\(k\)对,那么方案就是\(2^k\)。

怎么对于每个对称轴快速计算有多少对?也就是每对\(a\)都会对他们的对称轴产生贡献,假设\(i\)于\(j\)都是\(a\),那么对称轴就是\(\frac{i+j}{2}\),这个直接\(FFT\)就可以了

然后\(b\)同理。

然后字符串hash+二分计算连续的方案就好了

时间复杂度\(O(n\log n)\)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
#define ull unsigned long long
using namespace std;
const ll N=4e5+10,P=998244353,p=1e9+7;
const ull g=131;
ll n,l,a[N],b[N],r[N],ans;
ull pw[N],h[N],d[N];
char s[N];
ll power(ll x,ll b,ll P){
ll ans=1;
while(b){
if(b&1)ans=ans*x%P;
x=x*x%P;b>>=1;
}
return ans;
}
void NTT(ll *f,ll op){
for(ll i=0;i<n;i++)
if(i<r[i])swap(f[i],f[r[i]]);
for(ll p=2;p<=n;p<<=1){
ll tmp=power(3,(P-1)/p,P),len=(p>>1);
if(op==-1)tmp=power(tmp,P-2,P);
for(ll k=0;k<n;k+=p){
ll buf=1;
for(ll i=k;i<k+len;i++){
ll tt=buf*f[i+len]%P;
f[i+len]=(f[i]-tt+P)%P;
f[i]=(f[i]+tt)%P;
buf=buf*tmp%P;
}
}
}
if(op==-1){
ll invn=power(n,P-2,P);
for(ll i=0;i<n;i++)
f[i]=f[i]*invn%P;
}
return;
}
ull geth(ll l,ll r)
{return h[r]-h[l-1]*pw[r-l+1];}
ull getd(ll l,ll r)
{return d[l]-d[r+1]*pw[r-l+1];}
signed main()
{
scanf("%s",s);l=strlen(s);
for(ll i=0;i<l;i++)
a[i]=(s[i]=='a'),b[i]=(s[i]=='b');
n=1;while(n<2*l)n<<=1;
for(ll i=0;i<n;i++)r[i]=(r[i>>1]>>1)|((i&1)?(n>>1):0);
NTT(a,1);NTT(b,1);
for(ll i=0;i<n;i++)a[i]=a[i]*a[i]%P,b[i]=b[i]*b[i]%P;
NTT(a,-1);NTT(b,-1);
for(ll i=0;i<n;i++)a[i]=(a[i]+1)/2,b[i]=(b[i]+1)/2;
for(ll i=0;i<n;i++)
(ans+=power(2,a[i]+b[i],p)%p-1)%=p;
pw[0]=1;
for(ll i=1;i<=l;i++)pw[i]=pw[i-1]*g;
for(ll i=1;i<=l;i++)h[i]=h[i-1]*g+s[i-1]-'a';
for(ll i=l;i>=1;i--)d[i]=d[i+1]*g+s[i-1]-'a';
for(ll i=1;i<=l;i++){
ll L=1,R=min(i,l-i+1)-1;
while(L<=R){
ll mid=(L+R)>>1;
if(geth(i-mid,i+mid)==getd(i-mid,i+mid))L=mid+1;
else R=mid-1;
}
ans-=R+1;
}
for(ll i=1;i<l;i++){
ll L=1,R=min(i,l-i);
while(L<=R){
ll mid=(L+R)>>1;
if(geth(i-mid+1,i+mid)==getd(i-mid+1,i+mid))L=mid+1;
else R=mid-1;
}
ans-=R;
}
printf("%lld\n",(ans%p+p)%p);
return 0;
}

P4199-万径人踪灭【FFT】的更多相关文章

  1. P4199 万径人踪灭 FFT + manacher

    \(\color{#0066ff}{ 题目描述 }\) \(\color{#0066ff}{输入格式}\) 一行,一个只包含a,b两种字符的字符串 \(\color{#0066ff}{输出格式}\) ...

  2. 洛谷P4199 万径人踪灭(manacher+FFT)

    传送门 题目所求为所有的不连续回文子序列个数,可以转化为回文子序列数-回文子串数 回文子串manacher跑一跑就行了,考虑怎么求回文子序列数 我们考虑,如果$S_i$是回文子序列的对称中心,那么只要 ...

  3. BZOJ 3160: 万径人踪灭 [fft manacher]

    3160: 万径人踪灭 题意:求一个序列有多少不连续的回文子序列 一开始zz了直接用\(2^{r_i}-1\) 总-回文子串 后者用manacher处理 前者,考虑回文有两种对称形式(以元素/缝隙作为 ...

  4. bzoj 3160 万径人踪灭 FFT

    万径人踪灭 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 1936  Solved: 1076[Submit][Status][Discuss] De ...

  5. BZOJ 3160: 万径人踪灭 FFT+快速幂+manacher

    BZOJ 3160: 万径人踪灭 题目传送门 [题目大意] 给定一个长度为n的01串,求有多少个回文子序列? 回文子序列是指从原串中找出任意个,使得构成一个回文串,并且位置也是沿某一对称轴对称. 假如 ...

  6. 【bzoj3160】万径人踪灭 FFT

    题目:http://www.lydsy.com/JudgeOnline/problem.php?id=3160 我是一个傻叉 微笑脸 #include<bits/stdc++.h> #de ...

  7. Luogu4199 万径人踪灭 FFT、Manacher

    传送门 先不考虑”不是连续的一段“这一个约束条件.可以知道:第$i$位与第$j$位相同,可以对第$\frac{i+j}{2}$位置上产生$1$的贡献(如果$i+j$为奇数表明它会对一条缝产生$1$的贡 ...

  8. 【洛谷】P4199 万径人踪灭

    题解 每种字符跑一遍FFT,得到\(i + j = k\)时匹配的个数(要÷2,对于相同位置的最后再加上 然后算出\(2^{cnt[k]}\)的和,最后再减去用mancher匹配出的连续回文子串的个数 ...

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

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

  10. luogu P4199 万径人踪灭

    嘟嘟嘟 方案:回文子序列数 - 回文子串数. 回文子串数用manacher解决就行了,关键是怎么求会问序列数. 一个比较好的\(O(n ^ 2)\)的算法:对于一个回文中心\(i\),\(O(n)\) ...

随机推荐

  1. 教你IO流来便利电脑磁盘所有文件,把图片放到一个文件夹里(会发现什么不可告人的密码)

    一.需求 我要把C盘下面的所有图片都拿出来,放到一个新文件夹中.今天小编一身正气,看看有没有什么意外发现!!学会看看自己的盘,悄悄的哦!!! 二.代码展示(运行时间可能有点长) import java ...

  2. jsoup的Document类

    一.简介 Document是一个装载html的文档类,它是jsoup一个非常重要的类.类声明:public class Document extends Element .Document是Node间 ...

  3. 微信小程序学习笔记五 常见组件

    1. 常见组件 重点讲解小程序中常用的布局组件 1.1 view 代替 原来的div标签 <!-- pages/index/index.wxml --> <view hover-cl ...

  4. MFC中L, _T(),TEXT,_TEXT区别以及含义

    字符串前面加L表示该字符串是Unicode字符串. _T是一个宏,如果项目使用了Unicode字符集(定义了UNICODE宏),则自动在字符串前面加上L,否则字符串不变.因此,Visual C++里边 ...

  5. Linux - 安装 ant

    官方下载地址 https://ant.apache.org/bindownload.cgi 旧版下载地址 https://archive.apache.org/dist/ant/binaries/ 挑 ...

  6. java 使用匿名内部类实现多线程的创建

    匿名内部类的作用:简化代码 把子类继承父类,重写父类的方法,创建子类对象合一步完成 把实现类实现类接口,重写接口中的方法,创建实现类对象合成一步完成 匿名内部类的最终产物:子类/实现类对象,而这个类没 ...

  7. CSS导航菜单(二级菜单)

    index.html <div class="nav"> <ul> <li> <a href="#">Java& ...

  8. Vue获取Abp VNext Token

    Abp VNext默认没公开访问Token的Api,但有个问题Cookie方式如果是手机或桌面程序不如Token方便 Axios默认是Json方式提交,abp登录需要使用application/x-w ...

  9. 358 day09字节流、字符流

    day09[字节流.字符流] 主要内容 IO流 字节流 字符流 异常处理 Properties 教学目标 [ ] 能够说出IO流的分类和功能 [ ] 能够使用字节输出流写出数据到文件 [ ] 能够使用 ...

  10. php学习记录,使用script脚本

    echo "<script>alert()</script>"; 原来还能这么用,之前以为echo就是普通的用来打印 同时还可以在script标签下使用lo ...