hdu 4609 3-idiots <FFT>
链接: 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>的更多相关文章
- 解题:HDU 4609 Three Idiots
题面 要求组合的方法显然我们需要对桶卷积,即设$F(x)=\sum\limits_{i=1}^{maxx}x^{cnt[i]}$,然后我们初步的先把$F^2(x)$卷出来,表示选两条边.然后我们发现如 ...
- 快速傅里叶变换应用之二 hdu 4609 3-idiots
快速傅里叶变化有不同的应用场景,hdu4609就比较有意思.题目要求是给n个线段,随机从中选取三个,组成三角形的概率. 初始实在没发现这个怎么和FFT联系起来,后来看了下别人的题解才突然想起来:组合计 ...
- hdu 4609 3-idiots [fft 生成函数 计数]
hdu 4609 3-idiots 题意: 给出\(A_i\),问随机选择一个三元子集,选择的数字构成三角形的三边长的概率. 一开始一直想直接做.... 先生成函数求选两个的方案(注意要减去两次选择同 ...
- hdu 4609 3-idiots
http://acm.hdu.edu.cn/showproblem.php?pid=4609 FFT 不会 找了个模板 代码: #include <iostream> #include ...
- HDU 4609 3-idiots(FFT)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4609 题意:给出n个正整数(数组A).每次随机选出三个数.问这三个数能组成三角形的概率为多大? 思路: ...
- HDU 4609 FFT模板
http://acm.hdu.edu.cn/showproblem.php?pid=4609 题意:给你n个数,问任意取三边能够,构成三角形的概率为多少. 思路:使用FFT对所有长度的个数进行卷积(\ ...
- hdu 4609 3-idiots——FFT
题目:http://acm.hdu.edu.cn/showproblem.php?pid=4609 答案就是随便选三条边的方案 - 不合法的方案. 不合法的方案就是算出 x+y = k 的方案数 g[ ...
- hdu 4609 3-idiots —— FFT
题目:http://acm.hdu.edu.cn/showproblem.php?pid=4609 算不合法的比较方便: 枚举最大的边,每种情况算了2次,而全排列算了6次,所以还要乘3: 注意枚举最大 ...
- FFT(快速傅里叶变换):HDU 4609 3-idiots
3-idiots Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total S ...
随机推荐
- 009 The Interfaces In JAVA(官网文档翻译)
Interfaces There are a number of situations in software engineering when it is important for dispara ...
- svn提交自动同步到web目录
环境:CentOS6.5 测试线 svn和web服务器在同一台机器上 SVN版本1.6 情景:在测试环境下,开发部提出希望提交到SVN的代码能自动同步到WEB站点目录下,解决每次都要传双份这 ...
- Android开发-API指南-任务和回退栈
Task and Back Stack 英文原文: http://developer.android.com/guide/components/tasks-and-back-stack.html 采集 ...
- kubernetes学习笔记1
安装篇 Downloading Kubernetes You can either build a release from sources or download a pre-built relea ...
- 【SET】ORACLE SET 命令
set colsep' '; //-域输出分隔符 set echo off; //显示start启动的脚本中的每个sql命令,缺省为on set echo on ...
- [转载]word尾注插入参考文献——前人经验+自己总结
1. 以尾注的方式插入第一个参考文献. 将光标定位于word文档中将要插入参考文献的位置,按“插入/引用/脚注和尾注”.出现一菜单,选择“尾注”,“文档结尾”,编号格式为“1,2,3”.按“插入”按钮 ...
- 纯CSS气泡框实现方法探究
气泡框(或者提示框)是网页中一种很常见的元素,大多用来展示提示信息,如下图所示: 拆分来看,形如这种气泡框无外乎就是一个矩形框+一个指示方向的三角形小箭头,要制作出这样的气泡框,如果解决了三角形小箭头 ...
- c++ builder TreeView控件节点遍历
void __fastcall TForm1::GetRootNodes(TTreeView *DestTreeView)//得到所有根节点 { TTreeNode *vNode = NULL; vN ...
- AnyCAD三维控件场景节点详解
SceneNode是AnyCAD三维图形平台的AnyViz显示引擎的核心对象之一,只有放在场景管理器(SceneManager)里的节点才能被显示引擎所显示. 1. 节点分类 SceneNode是 ...
- 免费在线CAD文件转换
AnyCAD Exchange Cloud 提供在线的CAD文件转换服务,包括二维图纸和三维模型的数据转换. 支持的格式有: DWG/DGN/DXF 到 PDF, SVG, DAE等的转换 STEP/ ...