【题目链接】 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,将结果汇总,然后统计即可。

【代码】

  1. #include <cstdio>
  2. #include <cmath>
  3. #include <algorithm>
  4. #include <cstring>
  5. using namespace std;
  6. typedef long long LL;
  7. const int N=1048600;
  8. int n,pos[N];
  9. namespace FFT{
  10. struct comp{
  11. double r,i;
  12. comp(double _r=0,double _i=0):r(_r),i(_i){}
  13. comp operator +(const comp&x){return comp(r+x.r,i+x.i);}
  14. comp operator -(const comp&x){return comp(r-x.r,i-x.i);}
  15. comp operator *(const comp&x){return comp(r*x.r-i*x.i,i*x.r+r*x.i);}
  16. comp conj(){return comp(r,-i);}
  17. }A[N],B[N];
  18. const double pi=acos(-1.0);
  19. void FFT(comp a[],int n,int t){
  20. for(int i=1;i<n;i++)if(pos[i]>i)swap(a[i],a[pos[i]]);
  21. for(int d=0;(1<<d)<n;d++){
  22. int m=1<<d,m2=m<<1;
  23. double o=pi*2/m2*t;
  24. comp _w(cos(o),sin(o));
  25. for(int i=0;i<n;i+=m2){
  26. comp w(1,0);
  27. for(int j=0;j<m;j++){
  28. comp& A=a[i+j+m],&B=a[i+j],t=w*A;
  29. A=B-t;B=B+t;w=w*_w;
  30. }
  31. }
  32. }if(t==-1)for(int i=0;i<n;i++)a[i].r/=n;
  33. }
  34. }
  35. int l1,l2,ans[N],cnt=0,a[N],b[N];
  36. FFT::comp A[N],B[N],C[N];
  37. char s1[N],s2[N];
  38. int main(){
  39. scanf(" %s %s",&s1,&s2);
  40. l1=strlen(s1); l2=strlen(s2);
  41. for(int i=0;i<l1;i++)a[i]=s1[i]-'a'+1;
  42. for(int i=0;i<l2;i++)b[l2-1-i]=s2[i]=='?'?0:s2[i]-'a'+1;
  43. int N=1; while(N<l1+l2)N<<=1;
  44. int j=__builtin_ctz(N)-1;
  45. for(int i=0;i<N;i++){pos[i]=pos[i>>1]>>1|((i&1)<<j);}
  46. 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);
  47. FFT::FFT(A,N,1);FFT::FFT(B,N,1);
  48. for(int i=0;i<N;i++)C[i]=C[i]+A[i]*B[i];
  49. 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);
  50. FFT::FFT(A,N,1);FFT::FFT(B,N,1);
  51. for(int i=0;i<N;i++)C[i]=C[i]+A[i]*B[i];
  52. 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);
  53. FFT::FFT(A,N,1);FFT::FFT(B,N,1);
  54. for(int i=0;i<N;i++)C[i]=C[i]-A[i]*B[i]*FFT::comp(2,0);
  55. FFT::FFT(C,N,-1);
  56. for(int i=l2-1;i<l1;i++){
  57. if(C[i].r<0.5)ans[cnt++]=i-l2+1;
  58. }printf("%d\n",cnt);
  59. for(int i=0;i<cnt;i++)printf("%d\n",ans[i]);
  60. return 0;
  61. }

  

BZOJ 4503 两个串(FFT)的更多相关文章

  1. BZOJ 4503: 两个串 [FFT]

    4503: 两个串 题意:兔子们在玩两个串的游戏.给定两个只含小写字母的字符串S和T,兔子们想知道T在S中出现了几次, 分别在哪些位置出现.注意T中可能有"?"字符,这个字符可以匹 ...

  2. BZOJ.4503.两个串(FFT/bitset)

    题目链接 \(Description\) 给定两个字符串S和T,求T在S中出现了几次,以及分别在哪些位置出现.T中可能有'?'字符,这个字符可以匹配任何字符. \(|S|,|T|\leq 10^5\) ...

  3. bzoj 4503 两个串——FFT

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4503 翻转T,就变成卷积.要想想怎么判断. 因为卷积是乘积求和,又想到相等的话相减为0,所以 ...

  4. bzoj 4503 两个串 —— FFT

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4503 推式子即可: 不知怎的调了那么久,应该是很清晰的. 代码如下: #include< ...

  5. BZOJ 4503 两个串 ——FFT

    [题目分析] 定义两个字符之间的距离为 (ai-bi)^2*ai*bi 如果能够匹配,从i到i+m的位置的和一定为0 但这和暴力没有什么区别. 发现把b字符串反过来就可以卷积用FFT了. 听说KMP+ ...

  6. bzoj 4503 两个串

    Description 兔子们在玩两个串的游戏.给定两个字符串S和T,兔子们想知道T在S中出现了几次, 分别在哪些位置出现.注意T中可能有“?”字符,这个字符可以匹配任何字符. Input 两行两个字 ...

  7. 【刷题】BZOJ 4503 两个串

    Description 兔子们在玩两个串的游戏.给定两个字符串S和T,兔子们想知道T在S中出现了几次, 分别在哪些位置出现.注意T中可能有"?"字符,这个字符可以匹配任何字符. I ...

  8. bzoj 4503 两个串 快速傅里叶变换FFT

    题目大意: 给定两个\((length \leq 10^5)\)的字符串,问第二个串在第一个串中出现了多少次.并且第二个串中含有单字符通配符. 题解: 首先我们从kmp的角度去考虑 这道题从字符串数据 ...

  9. bzoj 4503: 两个串【脑洞+FFT】

    真实脑洞题 因为通配符所以导致t串实际有指数级别个,任何字符串相关算法都没有用 考虑一个新的匹配方法:设a串(模板串)长为n,从m串的i位置开始匹配:\( \sum_{i=0}^{n-1}(a[j]- ...

随机推荐

  1. MFC通过ODBC连接mysql(使用VS2012编写MFC)

    原创文章,转载请注明原文:MFC通过ODBC连接mysql(使用VS2012编写MFC) By Lucio.Yang 1.ODBC连接mysql 首先ODBC是什么呢? 开放数据库互连(Open Da ...

  2. Matlab图像直方图相关函数

    图像的灰度直方图(H是图像a.bmp的数据矩阵) imhist(H):%显示a的直方图 histeq(H); %将图像a进行直方图均衡化 adapthisteq(H); %将图像a进行直方图均衡化 i ...

  3. 测试通用的InsertOrUpdate

  4. 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 ...

  5. poj 2723 Get Luffy Out 二分+2-sat

    题目链接 给n个钥匙对, 每个钥匙对里有两个钥匙, 并且只能选择一个. 有m扇门, 每个门上有两个锁, 只要打开其中一个就可以通往下一扇门. 问你最多可以打开多少个门. 对于每个钥匙对, 如果选择了其 ...

  6. hdu 5637 Transform 最短路

    题目链接 异或的性质. 求s到t的最少步骤, 等价于求0到s^t的最少步骤. 通过最少的步骤达到s^t的状态, 等价于求0到s^t的最短路. 先将最短路求出来然后O(1)查询. #include &l ...

  7. CC++初学者编程教程(2) Microsoft Visual C++ 6.0开发环境搭建

    上一篇演示的是如何安装VS2010,本文演示的是如何安装Microsoft Visual C++ 6.0 简称VC6. 有同学经常VC6都是很古董的版本了,为啥他还存在,不得不说VC6是微软一个很经典 ...

  8. Windows Azure Web Role 的 IIS 重置

     如果您是一名 Web开发人员,您很可能使用过"简单快捷"的iisreset命令重置运行不正常的 IIS主机.这种方法通常在经典的 Windows Server VM上非常有效 ...

  9. jquery判断移动设备代码片段;pc、iphone、安卓

    $(document).ready(function () { /* 判断设备*/ var browser={ versions:function(){ var u = navigator.userA ...

  10. 高德地图API

    这周计划: 周一 早上 (高德地图API) 中午写(IFE PART ONE) 下午(高德地图API) 下班(IFE PART ONE)