zoukankan      html  css  js  c++  java
  • HDU 4609 3-idiots(FFT)

    【题目链接】 http://acm.hdu.edu.cn/showproblem.php?pid=4609

     

    【题目大意】

      给出一些数字,问从中随机选取三个数字,能够组成三角形的概率。

    【题解】

      首先,我们可以求出两个数字相加的集合,之后枚举最长边,求比其大的两边之和的个数就是这条边对答案的贡献值了。考虑求两个数字相加组合需要O(n2),复杂度难以承受,我们计算一个num数组,表示每个数字出现的次数,num数组的卷积结果就是每种两边之和的出现次数,FFT优化即可。

      在计算每条边的贡献的时候,注意要除去包括自己的两边之和,以及存在边比自己大的两边之和。

    【代码】

    #include <cstdio>
    #include <cmath>
    #include <algorithm>
    #include <cstring> 
    using namespace std;
    typedef long long LL;
    const int N=524300;
    int n,pos[N];
    namespace FFT{
        struct comp{
            double r,i;
            comp(double _r=0,double _i=0):r(_r),i(_i){}
            comp operator +(const comp&x){return comp(r+x.r,i+x.i);}
            comp operator -(const comp&x){return comp(r-x.r,i-x.i);}
            comp operator *(const comp&x){return comp(r*x.r-i*x.i,i*x.r+r*x.i);}
            comp conj(){return comp(r,-i);}
        }A[N],B[N];
        const double pi=acos(-1.0);
        void FFT(comp a[],int n,int t){
            for(int i=1;i<n;i++)if(pos[i]>i)swap(a[i],a[pos[i]]);
            for(int d=0;(1<<d)<n;d++){
                int m=1<<d,m2=m<<1;
                double o=pi*2/m2*t;
                comp _w(cos(o),sin(o));
                for(int i=0;i<n;i+=m2){
                    comp w(1,0);
                    for(int j=0;j<m;j++){
                        comp& A=a[i+j+m],&B=a[i+j],t=w*A;
                        A=B-t;
                        B=B+t;
                        w=w*_w;
                    }
                }
            }if(t==-1)for(int i=0;i<n;i++)a[i].r/=n;
        }
        void mul(long long *a,long long *b,long long *c,int k){
            int i,j;
            for(i=0;i<k;i++)A[i]=comp(a[i],b[i]);
            j=__builtin_ctz(k)-1;
            for(int i=0;i<k;i++){pos[i]=pos[i>>1]>>1|((i&1)<<j);} 
            FFT(A,k,1);
            for(int i=0;i<k;i++){
                j=(k-i)&(k-1);
                B[i]=(A[i]*A[i]-(A[j]*A[j]).conj())*comp(0,-0.25);
            }FFT(B,k,-1);
            for(int i=0;i<k;i++)c[i]=(long long)(B[i].r+0.5);
        }
    }int T,a[N];
    long long A[N],C[N],s[N];
    int main(){
        scanf("%d",&T);
        while(T--){
            scanf("%d",&n);
            memset(A,0,sizeof(A));
            for(int i=0;i<n;i++){
                scanf("%d",&a[i]);
                A[a[i]]++;
            }sort(a,a+n);
            int len=(a[n-1]+1)*2;
            int N=1; while(N<len)N<<=1;
            FFT::mul(A,A,C,N);
            for(int i=0;i<n;i++)C[a[i]+a[i]]--;
            for(int i=1;i<=len;i++)C[i]/=2;
            for(int i=1;i<=len;i++)s[i]=s[i-1]+C[i];
            long long cnt=0;
            for(int i=0;i<n;i++){
                cnt+=s[len]-s[a[i]];
                cnt-=(long long)(n-1-i)*i;
                cnt-=(n-1);
                cnt-=(long long)(n-1-i)*(n-2-i)/2;
            }long long tot=(long long)n*(n-1)*(n-2)/6;
            printf("%.7lf
    ",(double)cnt/tot);
        }return 0;
    }
    
  • 相关阅读:
    Android 四大组件学习之ContentProvider三
    JavaScript遍历table
    codecombat之KithGard地牢19-37关代码分享
    【学习笔记】信息系统项目管理-项目採购管理-合同分类
    【记中关村.西北食府.兰州拉面】诗一首
    HDU 1042.N!【高精度乘法】【8月24】
    Mac安装MySQL
    Best Time to Buy and Sell Stock I &amp;&amp; II &amp;&amp; III
    UVALive 6663 Count the Regions 离散+bfs染色_(:зゝ∠)_
    ftk学习记(combox篇)
  • 原文地址:https://www.cnblogs.com/forever97/p/hdu4609.html
Copyright © 2011-2022 走看看