【hdu4609】 3-idiots FFT

题外话:好久没写blog了啊~~

题目传送门

题目大意:给你m条长度为ai的线段,求在其中任选三条出来,能构成三角形的概率。即求在这n条线段中找出三条线段所能拼出的三角形数量除以$\binom{m}{3}$。

假设我们手中有3条长度分别为$x,y,z$的边(为了简化问题我们假设$x<y<z$,$x,y,z$相等的情况另行讨论),如果他们能拼成三角形,必然满足$x+y>z$且$z-y<x$。

该题的$O(m^3)$做法:枚举其中的3条边,套用上面的判断公式,进行累计。

但通过简单的变式,我们假设我们已经确定了$x$和$y$,那么$z$的范围即为$[y,x+y)$,我们维护一个数组$num$,$num_i$表示长度为i的线段数量。再维护一个num的前缀和sum。则$ans=\sum^{n-1}_{x=1}  \sum^{n}_{y=x+1} num_x * num_y*(sum[x+y-1]-sum[y])$,运算该和式的时间复杂度为$O(n^2)$,此处的n表示最长线段的长度。

然而还是会TLE。。。。

我们考虑对其做一些变式。

$ans=\sum^{n-1}_{x=1}  \sum^{n}_{y=x+1} num_x * num_y*(sum[x+y-1]-sum[y])$

 $=\sum^{n-1}_{x=1}  \sum^{n}_{y=x+1} num_x * num_y*sum[x+y-1] -\sum^{n-1}_{x=1}  \sum^{n}_{y=x+1} num_x * num_y*sum[y]$

该式子的后半部分,我们可以通过维护$num_y*sum[y]$的后缀和,实现O(n)的计算。

下面我们继续对式子的前半部分变式。令t=x+y。则有

$=\frac{1}{2}\sum^{2n}_{t=2}  \sum^{t-1}_{p=1} num_p*num_{t-p}*sum[t-1] -\frac{1}{2}\sum^{n}_{i=1}num_i^2*sum[2*i-1]$

我们发现,该式子的$\sum_{p=1}^{t-1} num_{p}\times num_{t-p}$,可以用FFT求出。则时间复杂度成功降低至O(n log n),而式子的后半部分可以O(n)求出,求和时间复杂度降低至O(n log n)。

下面说下x=y=z,x=y<z,x<y=z的处理方法。

x=y=z:$\sum _{\forall num_x>2} \binom{num_x}{3}$

x=y<z: $\sum _{x=1}^{n} \binom{num_x}{2}*(sum[2x-1]-sum[x])$

x<y=z:$\sum_{x=1}^{n}(sumc2_n-sumc2_x)$ ,其中$sumc2_x$ 表示$\sum_{i=1}^{x}\binom{num_x}{2}$。

把这三种情况和最初描述的情况相加即可。

 1 #include<bits/stdc++.h>
 2 #define M 270000
 3 #define cp complex<double>
 4 #define PI acos(-1)
 5 #define L long long
 6 using namespace std;
 7 cp a[M];
 8 void change(cp a[],int len){
 9     for(int i=0,j=0;i<len-1;i++){
10         if(i<j) swap(a[i],a[j]);
11         int k=len>>1;
12         while(j>=k) j-=k,k>>=1;
13         j+=k;
14     }
15 }
16 void fft(cp a[],int n,int on){
17     change(a,n); cp t,u;
18     for(int h=2;h<=n;h<<=1){
19         cp wn(cos(-on*2*PI/h),sin(-on*2*PI/h));
20         for(int j=0;j<n;j+=h){
21             cp w(1,0);
22             for(int k=j;k<j+(h>>1);k++){
23                 u=a[k]; t=w*a[k+(h>>1)];
24                 a[k]=u+t; a[k+(h>>1)]=u-t;
25                 w=w*wn;
26             }
27         }
28     }
29 }
30 L num[M]={0},sum[M]={0},sumhh[M]={0},sumc2[M]={0},sc[M]={0},ans=0;
31 
32 int Main(){
33     int m,n,maxn=0; scanf("%d",&n);
34     for(int i=1;i<=n;i++){
35         int x; scanf("%d",&x);
36         num[x]++; maxn=max(maxn,x);
37     }
38     for(m=1;m<(maxn*2+1);m<<=1);
39     for(int i=0;i<m;i++) a[i]=cp(num[i],0);
40     fft(a,m,1);
41     for(int i=0;i<m;i++) a[i]=a[i]*a[i];
42     fft(a,m,-1);
43     for(int i=0;i<m;i++) sc[i]=(a[i+1].real()+0.5)/m;
44     for(int i=1;i<m;i++){
45         sum[i]=sum[i-1]+num[i];
46         sumc2[i]=sumc2[i-1]+num[i]*(num[i]-1)/2;
47     }
48     
49     L ans1=0,ans2=0,ans3=0,ans4=0;
50     for(int i=1;i<=maxn;i++) if(num[i]>2) 
51     ans1+=(L)num[i]*(num[i]-1)*(num[i]-2);
52     ans1/=6;//x=y=z
53     for(int i=1;i<=maxn;i++) 
54     ans2+=(L)num[i]*(num[i]-1)*(sum[2*i-1]-sum[i]);
55     ans2/=2;//x=y<z
56     for(int i=1;i<=maxn;i++)
57     ans3+=num[i]*(sumc2[maxn]-sumc2[i]);
58     //x<y=z
59     for(int i=1;i<2*maxn;i++) ans4+=sum[i]*sc[i];
60     for(int i=1;i<=maxn;i++) ans4-=num[i]*num[i]*sum[2*i-1];
61     ans4/=2;//x<y<z卷积部分
62     
63     for(int i=maxn;i;i--) sumhh[i]=sumhh[i+1]+num[i]*sum[i];
64     for(int i=1;i<maxn;i++) ans4-=num[i]*sumhh[i+1];
65     ans=ans1+ans2+ans3+ans4;
66     
67     double fenmu=(L)n*(n-1)*(n-2)/6;
68     double hh=ans/fenmu;
69     printf("%.7lf\n",hh);
70 }
71 int main(){
72     //freopen("in.txt","r",stdin);
73     //freopen("out.txt","w",stdout);
74     int cas; scanf("%d",&cas);
75     while(cas--){
76         memset(num,0,sizeof(num)); memset(sum,0,sizeof(sum));
77         memset(sumhh,0,sizeof(sumhh)); memset(sumc2,0,sizeof(sumc2));
78         memset(sc,0,sizeof(sc)); memset(a,0,sizeof(a)); ans=0;
79         Main();
80     }
81 }

 

posted @ 2018-01-06 14:17  AlphaInf  阅读(192)  评论(0编辑  收藏  举报