bzoj4259 残缺的字符串

http://www.elijahqi.win/archives/3098
Description

很久很久以前,在你刚刚学习字符串匹配的时候,有两个仅包含小写字母的字符串A和B,其中A串长度为m,B串长度为n。可当你现在再次碰到这两个串时,这两个串已经老化了,每个串都有不同程度的残缺。
你想对这两个串重新进行匹配,其中A为模板串,那么现在问题来了,请回答,对于B的每一个位置i,从这个位置开始连续m个字符形成的子串是否可能与A串完全匹配?

Input

第一行包含两个正整数m,n(1<=m<=n<=300000),分别表示A串和B串的长度。
第二行为一个长度为m的字符串A。
第三行为一个长度为n的字符串B。
两个串均仅由小写字母和号组成,其中号表示相应位置已经残缺。

Output

第一行包含一个整数k,表示B串中可以完全匹配A串的位置个数。
若k>0,则第二行输出k个正整数,从小到大依次输出每个可以匹配的开头位置(下标从1开始)。

Sample Input

3 7
a*b
aebr*ob
Sample Output

2
1 5
HINT

Source

By Claris

orz fft 字符串匹配 好那么

如果把两个串’*’的部分当成0那么
有如下公式 i(a[i]b[i])2a[i]b[i]=0 ∑ i ( a [ i ] − b [ i ] ) 2 ∗ a [ i ] ∗ b [ i ] = 0 这个为什么一定要平方 因为有可能存在一些情况 比如 前面是负数然后后面是整数结果我这里恰好加成0了但是显然是不匹配的.. 然后针对这个式子把a反转一下 然后公式写出来化简一下发现其中分成三段每一段分别是卷积形式 所以就fft一下 最后看一下每位上是否为0 即可

#include<cmath>
#include<cstdio>
#include<cstring>
#define pi acos(-1)
#include<algorithm>
using namespace std;
const int N=300030;
struct C{
    double r,i;
    inline friend C operator +(const C &a,const C &b){return (C){a.r+b.r,a.i+b.i};}
    inline friend C operator -(const C &a,const C &b){return (C){a.r-b.r,a.i-b.i};}
    inline friend C operator *(const C &a,const C &b){return (C){a.r*b.r-a.i*b.i,a.r*b.i+a.i*b.r};}
    inline void operator *=(const C &a){*this=*this*a;}
}a[N<<2],b[N<<2],c[N<<2];
char ss1[N],s2[N];
int pend[N],ans,n,m,a1[N<<2],a2[N<<2],R[N<<2];
inline void fft(C *x,int f){
    for (int i=0;i<n;++i) if (i<R[i]) swap(x[i],x[R[i]]);
    for (int i=1;i<n;i<<=1){
        C wn=(C){cos(pi/i),f*sin(pi/i)};
        for (int j=0;j<n;j+=i<<1){
            C w=(C){1,0},t1,t2;
            for (int k=0;k<i;++k,w*=wn)
                t1=x[j+k],t2=x[j+i+k]*w,x[j+k]=t1+t2,x[j+i+k]=t1-t2;
        }
    }if (f==-1) for (int i=0;i<n;++i) x[i].r/=n;
}
int main(){
    freopen("bzoj4259.in","r",stdin);
    scanf("%d%d",&m,&n);scanf("%s",ss1);scanf("%s",s2);int nn=n;
    for (int i=0;i<m;++i) a1[i]=ss1[m-i-1]!='*'?ss1[m-i-1]-'a'+1:0;
    for (int i=0;i<n;++i) a2[i]=s2[i]!='*'?s2[i]-'a'+1:0;m+=n;int t=0;
    for (n=1;n<m;n<<=1,++t);for (int i=0;i<n;++i) R[i]=(R[i>>1]>>1)|(i&1)<<t-1;
    for (int i=0;i<n;++i) a[i]=(C){a1[i]*a1[i]*a1[i],0},b[i]=(C){a2[i],0};
    fft(a,1);fft(b,1);for (int i=0;i<n;++i) c[i]=c[i]+a[i]*b[i];m-=nn;
    for (int i=0;i<n;++i) a[i]=(C){a1[i]*a1[i],0},b[i]=(C){a2[i]*a2[i],0};
    fft(a,1);fft(b,1);for (int i=0;i<n;++i) c[i]=c[i]-(C){2,0}*a[i]*b[i];
    for (int i=0;i<n;++i) a[i]=(C){a1[i],0},b[i]=(C){a2[i]*a2[i]*a2[i],0};
    fft(a,1);fft(b,1);for (int i=0;i<n;++i) c[i]=c[i]+a[i]*b[i];fft(c,-1);
    for (int i=m-1;i<nn;++i) if (c[i].r<0.5) ++ans,pend[ans]=i+1-m+1;
    printf("%d\n",ans);for (int i=1;i<=ans;++i) printf("%d ",pend[i]);
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值