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

题意: 给定 N 个正整数, 表示 N 条线段的长度, 问任取 3 条, 可以构成三角形的概率为多少~

数据范围: N<=10^5 ~~

思路:设三边分别为 x, y, z (x<=y<=z) 枚举 z ,统计 x+y 大于 z 的数目 .

比赛时能想到的只有 O(n^2) 的算法,无力 AC~

赛后才知道有种东西叫 FFT ~

以下为官方解题报告:

/*

  记录 A_i 为长度为 i 的树枝的数量,并让 A 对它本身做 FFT,得到任意选两个树枝能得到的各个和的数量。枚举第三边,

计算出所有两边之和大于第三条边的方案数,并把前两条边包含最长边的情况减掉就是答案。

*/

设长度为 a1,a2, ....an, 统计每种长度个数为 cnt[ ai ] ; 可表示多项式      

多项式自乘后, 其指数即为 ai + aj 的值, 系数即为 方案数, 减去不合法的即为所求~

 #include <cstdio>
#include <cmath>
#include <iostream>
#include <algorithm>
#include <cstring>
using namespace std;
typedef __int64 LL;
const double pi=acos(-);
const int MAXN=3e5;
const double eps=1e-;
struct C
{
double r, i;
C (){}
C(double _r, double _i):r(_r),i(_i){}
inline C operator +(const C a)const{
return C(r+a.r, i+a.i);
}
inline C operator - (const C a)const {
return C(r-a.r, i-a.i);
}
inline C operator * (const C a)const{
return C(r*a.r-i*a.i, r*a.i+i*a.r);
}
}a[MAXN], b[MAXN];
int num[MAXN], cnt[MAXN];
LL res[MAXN], sum[MAXN]; void brc(C *y,int l) // 二进制平摊反转置换 O(logn)
{
register int i,j,k;
for(i=,j=l>>;i<l-;i++){
if(i<j) swap(y[i],y[j]); // 交换互为下标反转的元素
// i<j保证只交换一次
k=l>>;
while(j>=k) {// 由最高位检索,遇1变0,遇0变1,跳出
j-=k;
k>>=;
}
if(j<k) j+=k;
}
}
void FFT(C *y,int l,int on) // FFT O(nlogn)
// 其中on==1时为DFT,on==-1为IDFT
{
register int h,i,j,k;
C u,t;
brc(y,l); // 调用反转置换
for(h=;h<=l;h<<=) // 控制层数
{
// 初始化单位复根
C wn(cos(on**pi/h),sin(on**pi/h));
for(j=;j<l;j+=h) // 控制起始下标
{
C w(,); // 初始化螺旋因子
for(k=j;k<j+h/;k++) // 配对
{
u=y[k];
t=w*y[k+h/];
y[k]=u+t;
y[k+h/]=u-t;
w=w*wn; // 更新螺旋因子
} // 据说上面的操作叫蝴蝶操作…
}
}
if(on==-) for(i=;i<l;i++) y[i].r/=l; // IDFT
} int n, L , Max, T;
int main() {
scanf("%d", &T);
while (T--) {
Max = ;
memset(cnt, , sizeof(cnt));
scanf("%d", &n);
for (int i = ; i < n; ++i) {
scanf("%d", num + i);
cnt[num[i]]++;
Max = max(Max, num[i]);
}
++Max;L = ;
while (L < Max <<) {
L <<= ;
}
for (int i = ; i < Max; ++i) {
a[i] = C(cnt[i], );
} for (int i = Max; i < L; ++i) {
a[i] = C(, );
}
FFT(a, L, );
for (int i = ; i < L; ++i)
a[i] = a[i] * a[i]; // 多项式自乘
FFT(a, L, -); for (int i = ; i < L; ++i) {
res[i] = (LL) (a[i].r + 0.5);
} for (int i = ; i <= Max; ++i)
res[i << ] -= cnt[i];// 自己和自己乘, 即 枚举的是 x+x 的 for (int i = ; i < L; ++i)
res[i] >>= ; // x+y 与 y+x 算一次
for (int i = ; i < L; ++i) { //前缀和
sum[i] = sum[i - ] + res[i];
}
double tot = , den = 1.0*n * (n - ) * (n - )/; for (int i = ; i < n; ++i) {
tot += sum[num[i]] / den;// x+y<=z 的个数 }
double ans = - tot ;
printf("%.7f\n", ans);
} return ;
}

hdu 4609 3-idiots <FFT>的更多相关文章

  1. 解题:HDU 4609 Three Idiots

    题面 要求组合的方法显然我们需要对桶卷积,即设$F(x)=\sum\limits_{i=1}^{maxx}x^{cnt[i]}$,然后我们初步的先把$F^2(x)$卷出来,表示选两条边.然后我们发现如 ...

  2. 快速傅里叶变换应用之二 hdu 4609 3-idiots

    快速傅里叶变化有不同的应用场景,hdu4609就比较有意思.题目要求是给n个线段,随机从中选取三个,组成三角形的概率. 初始实在没发现这个怎么和FFT联系起来,后来看了下别人的题解才突然想起来:组合计 ...

  3. hdu 4609 3-idiots [fft 生成函数 计数]

    hdu 4609 3-idiots 题意: 给出\(A_i\),问随机选择一个三元子集,选择的数字构成三角形的三边长的概率. 一开始一直想直接做.... 先生成函数求选两个的方案(注意要减去两次选择同 ...

  4. hdu 4609 3-idiots

    http://acm.hdu.edu.cn/showproblem.php?pid=4609 FFT  不会 找了个模板 代码: #include <iostream> #include ...

  5. HDU 4609 3-idiots(FFT)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4609 题意:给出n个正整数(数组A).每次随机选出三个数.问这三个数能组成三角形的概率为多大? 思路: ...

  6. HDU 4609 FFT模板

    http://acm.hdu.edu.cn/showproblem.php?pid=4609 题意:给你n个数,问任意取三边能够,构成三角形的概率为多少. 思路:使用FFT对所有长度的个数进行卷积(\ ...

  7. hdu 4609 3-idiots——FFT

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=4609 答案就是随便选三条边的方案 - 不合法的方案. 不合法的方案就是算出 x+y = k 的方案数 g[ ...

  8. hdu 4609 3-idiots —— FFT

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=4609 算不合法的比较方便: 枚举最大的边,每种情况算了2次,而全排列算了6次,所以还要乘3: 注意枚举最大 ...

  9. FFT(快速傅里叶变换):HDU 4609 3-idiots

    3-idiots Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

随机推荐

  1. 详细讲解JAVA中的IO流

    一.流的概念        流(stream)的概念源于UNIX中管道(pipe)的概念.在UNIX中,管道是一条不间断的字节流,用来实现程序或进程间的通信,或读写外围设备.外部文件等.        ...

  2. linux 内存使用

    # df -h Filesystem Size Used Avail Use% Mounted on /dev/sda1 50G 1.9G 45G 5% / tmpfs 1.9G 0 1.9G 0% ...

  3. AX 获得当前Grid的数据源的记录行数

    sysQuery::CountTotal();方法 eg: int lines = sysQuery::CountTotal( SalesTable_ds.QueryRun());

  4. C# 排列组合

    排列组合的概念 排列:从n个不同元素中取出m(m≤n)个元素,按照一定的顺序排成一列,叫做从n个元素中取出m个元素的一个排列(Arrangement). 组合:从m个不同的元素中,任取n(n≤m)个元 ...

  5. idea 14 svn安装

    安装SVN客户端. 打开IDEA 14 File-setting-Version coltorl-Subversion-General 填入安装路径 打开"VCS"菜单项然后点击& ...

  6. C# 加密算法

     public static class Common     {         #region MD5加密         /// <summary>            /// M ...

  7. Volume serial number could associate file existence on certain volume

    When it comes to lnk file analysis, we should put more emphasis on the volume serial number. It coul ...

  8. UIButton设置imgae图片自适应button的大小且不变形

    在某些情况下,我们使用的UIButton的背景图片不一定就是标准的尺寸,有时会偏大,那么怎么办? 这个比较简单直接设置 :    self.imageView.contentMode = UIView ...

  9. 一款安卓ShowcaseView视图源码效果

    该源码是从源码天堂那边转载过来的,大家可以看看一下吧啊,一款安卓ShowcaseView视图源码效果,非常不错的,特别是在做引导时使用. 源码下载地址:http://code.662p.com/vie ...

  10. 两张table数据同步--使用触发器

    数据同步, 如果每天同步一次的话可以使用SSIS,跑JOB等,可以同步不同的DB的数据: 实时的可以使用触发器,在同一个DB中(或者DB Link): USE [test] GO IF EXISTS( ...