[BZOJ]3527 力(ZJOI2014)

  第一次背出FFT模板,在此mark一道裸題。學習

Description

  給出n個數qi,給出Fj的定義以下:spa

  

  令Ei=Fi/qi,求Ei。code

Input

  第一行一個整數n。
  接下來n行每行輸入一個數,第i行表示qi。

Output

  n行,第i行輸出Ei。與標準答案偏差不超過1e-2便可。orm

Sample Input

  5
  4006373.885184
  15375036.435759
  1717456.469144
  8514941.004912
  1410681.345880blog

Sample Output

  -16838672.693
  3439.793
  7509018.566
  4595686.886
  10903040.872ip

HINT

  n ≤ 100000,0 < qi < 1000000000。get

 

Solution

  看到題目中 下標爲j的項等於下標爲i的項與下標爲j±i的項的乘積之和,你應該會有所感受吧。string

  設,那麼 it

  顯然兩邊都是卷積的式子,因此兩邊分別作一次FFT就能夠了。io

  然而咱們再思考一下,發現兩邊的式子是能夠合併的:

  設,那麼 就徹底成立了。只要作一次FFT就夠了。

  時間複雜度O(nlogn)。

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#define pi acos(-1)
#define MN 263005
using namespace std;
struct cp
{
    double v,i;
    friend cp operator+(const cp& a,const cp& b) {return (cp){a.v+b.v,a.i+b.i};}
    friend cp operator-(const cp& a,const cp& b) {return (cp){a.v-b.v,a.i-b.i};}
    friend cp operator*(const cp& a,const cp& b) {return (cp){a.v*b.v-a.i*b.i,a.v*b.i+a.i*b.v};}
}w[2][MN],A[MN],B[MN],C[MN];
double a[MN];
int r[MN];
int N,n;

void init(int n)
{
    register int i,j,k;
    for (N=1;N<=n;N<<=1); cp g=(cp){cos(pi*2/N),sin(pi*2/N)};
    for (i=j=0;i<N;r[++i]=j)
        for (k=N>>1;(j^=k)<k;k>>=1);
    w[0][0]=w[1][0]=(cp){1,0};
    for (i=1;i<N;++i) w[0][i]=w[0][i-1]*g;
    for (i=1;i<N;++i) w[1][i]=w[0][N-i];
}

void FFT(cp* a,bool g)
{
    register int i,j,k;
    for (i=1;i<N;++i) if (r[i]<i) swap(a[i],a[r[i]]);
    for (i=1;i<N;i<<=1)
        for (j=0;j<N;j+=(i<<1))
            for (k=0;k<i;++k)
            {
                cp x=a[i+j+k]*w[g][N/(i<<1)*k];
                a[i+j+k]=a[j+k]-x;
                a[j+k]=a[j+k]+x;
            }
    if (g) for (i=0;i<N;++i) a[i].v/=N,a[i].i/=N;
}

int main()
{
    register int i;
    scanf("%d",&n);
    for (i=1;i<=n;++i) scanf("%lf",&a[i]),A[i].v=a[i];
    for (i=1;i<n;++i)
        B[n+i].v=(double)1/i/i,B[n-i].v=(double)-1/i/i;
    init(n<<1);
    FFT(A,0); FFT(B,0);
    for (i=0;i<N;++i) C[i]=A[i]*B[i];
    FFT(C,1);
    for (i=1;i<=n;++i) printf("%.7lf\n",C[n+i].v);
}

 

Last Word

  推薦miskcoo的關於學習FFT的blog:從多項式乘法到快速傅里葉變換

相關文章
相關標籤/搜索