zoukankan      html  css  js  c++  java
  • BZOJ 4259 残缺的字符串 ——FFT

    【题目分析】

        同bzoj4503。

        只是精度比较卡,需要试一试才能行O(∩_∩)O

        用过long double,也加过0.4。最后发现判断的时候改成0.4就可以了

    【代码】

    #include <cmath>
    #include <cstdio>
    #include <cstring>
    #include <iostream>
    #include <algorithm>
    using namespace std;
    
    #define maxn 1200005
    #define F(i,j,k) for (int i=j;i<=k;++i)
    #define D(i,j,k) for (int i=j;i>=k;--i)
    
    const double pi=acos(-1.0);
    
    struct Complex{
    	double x,y;
    	Complex operator + (Complex a){Complex b; return b.x=x+a.x,b.y=y+a.y,b;}
    	Complex operator - (Complex a){Complex b; return b.x=x-a.x,b.y=y-a.y,b;}
    	Complex operator * (Complex a){Complex b; return b.x=x*a.x-y*a.y,b.y=x*a.y+y*a.x,b;}
    }a[maxn],b[maxn],c[maxn];
    
    int A[maxn],B[maxn],la,lb,n,m=1,len,rev[maxn],ans[maxn],cnt=0;
    char s1[maxn],s2[maxn];
    
    void FFT(Complex * x,int n,int f)
    {
    	F(i,0,n-1) if (rev[i]>i) swap(x[rev[i]],x[i]);
    	for (int m=2;m<=n;m<<=1)
    	{
    		int mid=m>>1;
    		Complex wn; wn.x=cos(2.0*pi/m*f); wn.y=sin(2.0*pi/m*f);
    		for (int i=0;i<n;i+=m)
    		{
    			Complex w; w.x=1.0; w.y=0;
    			F(j,0,mid-1)
    			{
    				Complex a=x[i+j],b=x[i+j+mid]*w;
    				x[i+j]=a+b; x[i+j+mid]=a-b;
    				w=w*wn;
    			}
    		}
    	}
    }
    
    int main()
    {
    //	freopen("in.txt","r",stdin);
    	scanf("%d%d",&lb,&la);
    	scanf("%s",s2);
    	scanf("%s",s1);
    	n=la+lb+1;
    	while (m<=n) m<<=1,len++; n=m;
    	F(i,0,n-1)
    	{
    		int t=i,ret=0;
    		F(j,1,len) ret<<=1,ret|=t&1,t>>=1;
    		rev[i]=ret;
    	}
    	F(i,0,la-1)	{A[i]=s1[i]-'a'+1; if (s1[i]=='*') A[i]=0;}
    	F(i,0,lb-1)	{B[i]=s2[lb-1-i]-'a'+1;	if (s2[lb-1-i]=='*') B[i]=0;}
    
    	F(i,0,n-1) a[i].x=A[i]*A[i]*A[i],a[i].y=0;
    	F(i,0,n-1) b[i].x=B[i],b[i].y=0;
    	FFT(a,n,1); FFT(b,n,1);
    	F(i,0,n-1) c[i]=a[i]*b[i];
    	
    	F(i,0,n-1) a[i].x=A[i],a[i].y=0;
    	F(i,0,n-1) b[i].x=B[i]*B[i]*B[i],b[i].y=0;
    	FFT(a,n,1); FFT(b,n,1);
    	F(i,0,n-1) c[i]=c[i]+a[i]*b[i];
    	
    	F(i,0,n-1) a[i].x=2*A[i]*A[i],a[i].y=0;
    	F(i,0,n-1) b[i].x=B[i]*B[i],b[i].y=0;
    	FFT(a,n,1); FFT(b,n,1);
    	F(i,0,n-1) c[i]=c[i]-a[i]*b[i];
    	
    	FFT(c,n,-1);
    	F(i,0,n-1) c[i].x=c[i].x/n;
    //	F(i,0,n-1) printf("%.3f ",c[i].x); printf("
    ");
    	F(i,lb-1,la-1) if (fabs(c[i].x)<0.4) ans[++cnt]=i-lb+2;
    	printf("%d
    ",cnt);
    	F(i,1,cnt) printf("%d%c",ans[i],i==cnt?'
    ':' ');
    }
    

      

  • 相关阅读:
    java 学习进度二
    java 学习进度一
    Lua 笔记17
    chrome浏览器扩展的事件处理
    指定的参数错误。Vim.Host.DiskPartitionInfo.-spec VSPHERE.LOCALAdministrator WIN-DOPGQVRRU2C
    MySQL分区表
    MySQL绿色版的安装(mysql-5.6.22-win32.zip)
    Oracle操作语言分类
    注册表法修改IE8安全级别的方法
    JDK版本不兼容问题之:一台机器安装多个版本的JDK
  • 原文地址:https://www.cnblogs.com/SfailSth/p/6395892.html
Copyright © 2011-2022 走看看