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 ...
随机推荐
- 解决 Timeout expired. The timeout period elapsed prior to completion of the operation or the server is not responding. 的问题
在web 网站开发中,经常需要连接数据库,有时候会出现这样的数据连接异常消息: 主要原因是 应用程序与数据库的连接超出了数据库连接的默认时长,在这种情况下,我们可以把数据库连接的时长延长一些,因为 C ...
- 解决spawn-fcgi child exited with: 1
spawn-fcgi -d /data/web/ad/ -f /data/web/ad/code.py -a -P /data/openresty_81/nginx/pid/ad.pid 出错的时候请 ...
- 学习总结 java 输入输出流
思维导图 代码实际演示 package com.hanqi.io; import java.io.*; public class Test1 { public static void main(Str ...
- K均值算法实现
运行环境:Ubuntu+Code::Blocks(G++) K-均值:在D(数据集)中随机地选择k个对象,每个对象代表一个簇的初始均值或中心.对剩下的每个对象,根据其与各个簇中心的欧式距离,将它分配到 ...
- coffeeScript 语法总结
CoffeeScript ---->安装node.js ---->安装coffeeScript 语句: 注意:没有分号,语句由新的一行结束:多条语句写到同一行时需要分号表示一条语句的结束( ...
- Leetcode 0025. Reverse Nodes in k-Group
居然把头插法写错了,debug了一个多小时 /** * Definition for singly-linked list. * struct ListNode { * int val; * List ...
- postgresql安装配置
一,什么是postgresql PostgreSQL是以加州大学伯克利分校计算机系开发的 POSTGRES 版本 4.2 为基础的对象关系型数据库管理系统(ORDBMS),简称pgsql,它支持大部分 ...
- CSS 之 Opacity多浏览器透明度兼容处理
用来设定元素透明度的 Opacity 是CSS 3里的一个属性.当然现在还只有少部分浏览器支持. 不过各个浏览器都有自己的私有属性来支持,其中包括老版本的Mozilla和Safari: IE: fil ...
- JS常用的设计模式(9)——策略模式
策略模式的意义是定义一系列的算法,把它们一个个封装起来,并且使它们可相互替换.一个小例子就能让我们一目了然. 回忆下jquery里的animate方法. $( div ).animate( {&quo ...
- 重温CSS:Border属性
边界是众所周知的,有什么新的东西吗?好吧,我敢打赌,在这篇文章中,有很多你不看永远不知道的东西! 不仅可以用CSS3来创建圆角,使用原有CSS一样可以显示自定义图形.这是正确的(有待考究):在过去,没 ...