BZOJ 4503 两个串(FFT)
【题目链接】 http://www.lydsy.com/JudgeOnline/problem.php?id=4503
【题目大意】
给出S串和T串,计算T在S中出现次数,T中有通配符'?'。
【题解】
我们定义f[x]=sum_{i=0}^{n-1}|s1[i]-s2[i]|,当f[x]=0时,两个字符串相等。因为考虑到这里还有适配符,所以用f[x]=sum_{i=0}^{n-1}(s1[i]-s2[i])*(s1[i]-s2[i])*s1[i]*s2[i]来表示匹配函数。我们可以发现,如果将一个串倒置,那么这就是一个卷积的式子。因此我们将多项式展开,将得到的相加的三段式子,做三次FFT,将结果汇总,然后统计即可。
【代码】
- #include <cstdio>
- #include <cmath>
- #include <algorithm>
- #include <cstring>
- using namespace std;
- typedef long long LL;
- const int N=1048600;
- int n,pos[N];
- namespace FFT{
- struct comp{
- double r,i;
- comp(double _r=0,double _i=0):r(_r),i(_i){}
- comp operator +(const comp&x){return comp(r+x.r,i+x.i);}
- comp operator -(const comp&x){return comp(r-x.r,i-x.i);}
- comp operator *(const comp&x){return comp(r*x.r-i*x.i,i*x.r+r*x.i);}
- comp conj(){return comp(r,-i);}
- }A[N],B[N];
- const double pi=acos(-1.0);
- void FFT(comp a[],int n,int t){
- for(int i=1;i<n;i++)if(pos[i]>i)swap(a[i],a[pos[i]]);
- for(int d=0;(1<<d)<n;d++){
- int m=1<<d,m2=m<<1;
- double o=pi*2/m2*t;
- comp _w(cos(o),sin(o));
- for(int i=0;i<n;i+=m2){
- comp w(1,0);
- for(int j=0;j<m;j++){
- comp& A=a[i+j+m],&B=a[i+j],t=w*A;
- A=B-t;B=B+t;w=w*_w;
- }
- }
- }if(t==-1)for(int i=0;i<n;i++)a[i].r/=n;
- }
- }
- int l1,l2,ans[N],cnt=0,a[N],b[N];
- FFT::comp A[N],B[N],C[N];
- char s1[N],s2[N];
- int main(){
- scanf(" %s %s",&s1,&s2);
- l1=strlen(s1); l2=strlen(s2);
- for(int i=0;i<l1;i++)a[i]=s1[i]-'a'+1;
- for(int i=0;i<l2;i++)b[l2-1-i]=s2[i]=='?'?0:s2[i]-'a'+1;
- int N=1; while(N<l1+l2)N<<=1;
- int j=__builtin_ctz(N)-1;
- for(int i=0;i<N;i++){pos[i]=pos[i>>1]>>1|((i&1)<<j);}
- for(int i=0;i<N;i++)A[i]=FFT::comp(a[i]*a[i]*a[i],0),B[i]=FFT::comp(b[i],0);
- FFT::FFT(A,N,1);FFT::FFT(B,N,1);
- for(int i=0;i<N;i++)C[i]=C[i]+A[i]*B[i];
- for(int i=0;i<N;i++)A[i]=FFT::comp(a[i],0),B[i]=FFT::comp(b[i]*b[i]*b[i],0);
- FFT::FFT(A,N,1);FFT::FFT(B,N,1);
- for(int i=0;i<N;i++)C[i]=C[i]+A[i]*B[i];
- for(int i=0;i<N;i++)A[i]=FFT::comp(a[i]*a[i],0),B[i]=FFT::comp(b[i]*b[i],0);
- FFT::FFT(A,N,1);FFT::FFT(B,N,1);
- for(int i=0;i<N;i++)C[i]=C[i]-A[i]*B[i]*FFT::comp(2,0);
- FFT::FFT(C,N,-1);
- for(int i=l2-1;i<l1;i++){
- if(C[i].r<0.5)ans[cnt++]=i-l2+1;
- }printf("%d\n",cnt);
- for(int i=0;i<cnt;i++)printf("%d\n",ans[i]);
- return 0;
- }
BZOJ 4503 两个串(FFT)的更多相关文章
- BZOJ 4503: 两个串 [FFT]
4503: 两个串 题意:兔子们在玩两个串的游戏.给定两个只含小写字母的字符串S和T,兔子们想知道T在S中出现了几次, 分别在哪些位置出现.注意T中可能有"?"字符,这个字符可以匹 ...
- BZOJ.4503.两个串(FFT/bitset)
题目链接 \(Description\) 给定两个字符串S和T,求T在S中出现了几次,以及分别在哪些位置出现.T中可能有'?'字符,这个字符可以匹配任何字符. \(|S|,|T|\leq 10^5\) ...
- bzoj 4503 两个串——FFT
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4503 翻转T,就变成卷积.要想想怎么判断. 因为卷积是乘积求和,又想到相等的话相减为0,所以 ...
- bzoj 4503 两个串 —— FFT
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4503 推式子即可: 不知怎的调了那么久,应该是很清晰的. 代码如下: #include< ...
- BZOJ 4503 两个串 ——FFT
[题目分析] 定义两个字符之间的距离为 (ai-bi)^2*ai*bi 如果能够匹配,从i到i+m的位置的和一定为0 但这和暴力没有什么区别. 发现把b字符串反过来就可以卷积用FFT了. 听说KMP+ ...
- bzoj 4503 两个串
Description 兔子们在玩两个串的游戏.给定两个字符串S和T,兔子们想知道T在S中出现了几次, 分别在哪些位置出现.注意T中可能有“?”字符,这个字符可以匹配任何字符. Input 两行两个字 ...
- 【刷题】BZOJ 4503 两个串
Description 兔子们在玩两个串的游戏.给定两个字符串S和T,兔子们想知道T在S中出现了几次, 分别在哪些位置出现.注意T中可能有"?"字符,这个字符可以匹配任何字符. I ...
- bzoj 4503 两个串 快速傅里叶变换FFT
题目大意: 给定两个\((length \leq 10^5)\)的字符串,问第二个串在第一个串中出现了多少次.并且第二个串中含有单字符通配符. 题解: 首先我们从kmp的角度去考虑 这道题从字符串数据 ...
- bzoj 4503: 两个串【脑洞+FFT】
真实脑洞题 因为通配符所以导致t串实际有指数级别个,任何字符串相关算法都没有用 考虑一个新的匹配方法:设a串(模板串)长为n,从m串的i位置开始匹配:\( \sum_{i=0}^{n-1}(a[j]- ...
随机推荐
- MFC通过ODBC连接mysql(使用VS2012编写MFC)
原创文章,转载请注明原文:MFC通过ODBC连接mysql(使用VS2012编写MFC) By Lucio.Yang 1.ODBC连接mysql 首先ODBC是什么呢? 开放数据库互连(Open Da ...
- Matlab图像直方图相关函数
图像的灰度直方图(H是图像a.bmp的数据矩阵) imhist(H):%显示a的直方图 histeq(H); %将图像a进行直方图均衡化 adapthisteq(H); %将图像a进行直方图均衡化 i ...
- 测试通用的InsertOrUpdate
- Could not lock surface java.lang.IllegalArgumentException
08-07 14:46:33.795: E/Surface(4927): dequeueBuffer failed (Invalid argument) 08-07 14:46:33.800: E/V ...
- poj 2723 Get Luffy Out 二分+2-sat
题目链接 给n个钥匙对, 每个钥匙对里有两个钥匙, 并且只能选择一个. 有m扇门, 每个门上有两个锁, 只要打开其中一个就可以通往下一扇门. 问你最多可以打开多少个门. 对于每个钥匙对, 如果选择了其 ...
- hdu 5637 Transform 最短路
题目链接 异或的性质. 求s到t的最少步骤, 等价于求0到s^t的最少步骤. 通过最少的步骤达到s^t的状态, 等价于求0到s^t的最短路. 先将最短路求出来然后O(1)查询. #include &l ...
- CC++初学者编程教程(2) Microsoft Visual C++ 6.0开发环境搭建
上一篇演示的是如何安装VS2010,本文演示的是如何安装Microsoft Visual C++ 6.0 简称VC6. 有同学经常VC6都是很古董的版本了,为啥他还存在,不得不说VC6是微软一个很经典 ...
- Windows Azure Web Role 的 IIS 重置
如果您是一名 Web开发人员,您很可能使用过"简单快捷"的iisreset命令重置运行不正常的 IIS主机.这种方法通常在经典的 Windows Server VM上非常有效 ...
- jquery判断移动设备代码片段;pc、iphone、安卓
$(document).ready(function () { /* 判断设备*/ var browser={ versions:function(){ var u = navigator.userA ...
- 高德地图API
这周计划: 周一 早上 (高德地图API) 中午写(IFE PART ONE) 下午(高德地图API) 下班(IFE PART ONE)