题解:

好神的一道题。蒟蒻只能膜拜题解。

考虑a对b的贡献,如果a是a-b路径上第一个删除的点,那么给b贡献1。

所以转化之后就是求sigma(1/dist(i,j)),orz!!!

如果不是分母的话O(n)就可以搞,但是现在在分母上。。。

考虑转化一下,求ret[i]表示距离为i的点对有多少对。我们发现只要求出ret数组,然后就可以回答了。

如何求ret,我们用点分治。类似于RACE那道题。

对于一颗子树,我们整个信息一块统计,让它和前面的所有做卷积,更新ret,然后再把这棵子树归入前面的信息内。

代码:

 #include<cstdio>

 #include<cstdlib>

 #include<cmath>

 #include<cstring>

 #include<algorithm>

 #include<iostream>

 #include<vector>

 #include<map>

 #include<set>

 #include<queue>

 #include<string>

 #define inf 1000000000

 #define maxn 50000+5

 #define maxm 20000000+5

 #define eps 1e-10

 #define ll long long

 #define pa pair<int,int>

 #define for0(i,n) for(int i=0;i<=(n);i++)

 #define for1(i,n) for(int i=1;i<=(n);i++)

 #define for2(i,x,y) for(int i=(x);i<=(y);i++)

 #define for3(i,x,y) for(int i=(x);i>=(y);i--)
#define for4(i,x) for(int i=head[x],y;i;i=e[i].next) #define mod 1000000007 using namespace std; inline int read() { int x=,f=;char ch=getchar(); while(ch<''||ch>''){if(ch=='-')f=-;ch=getchar();} while(ch>=''&&ch<=''){x=*x+ch-'';ch=getchar();} return x*f; }
int n,mx,m,len,cnt,sum,tot,rt,rev[maxn],head[maxn],d[maxn],f[maxn],s[maxn],g[maxn];
bool del[maxn];
ll ret[maxn];
const double PI=acos(-1.0);
struct cp
{
double x,y;
cp operator +(cp b){return (cp){x+b.x,y+b.y};}
cp operator -(cp b){return (cp){x-b.x,y-b.y};}
cp operator *(cp b){return (cp){x*b.x-y*b.y,x*b.y+y*b.x};}
};
cp a[*maxn],b[*maxn],c[*maxn],y[*maxn];
struct edge{int go,next;}e[*maxn];
inline void insert(int x,int y)
{
e[++tot]=(edge){y,head[x]};head[x]=tot;
e[++tot]=(edge){x,head[y]};head[y]=tot;
}
inline void getrt(int x,int fa)
{
s[x]=;f[x]=;
for4(i,x)if(!del[y=e[i].go]&&y!=fa)
{
getrt(y,x);
s[x]+=s[y];
f[x]=max(f[x],s[y]);
}
f[x]=max(f[x],sum-f[x]);
if(f[x]<f[rt])rt=x;
}
inline void getdep(int x,int fa,int w)
{
if(w>mx)mx=w;
d[++cnt]=w;
for4(i,x)if(!del[y=e[i].go]&&y!=fa)getdep(y,x,w+);
}
inline void get(int n)
{
n++;n=*n-;
m=,len=;
while(m<=n)m<<=,len++;
for0(i,m-)
{
int x=i,y=;
for1(j,len)y<<=,y|=x&,x>>=;
rev[i]=y;
}
}
inline void fft(cp *x,int n,int flag)
{
for0(i,n-)y[rev[i]]=x[i];
for0(i,n-)x[i]=y[i];
for(int m=;m<=n;m<<=)
{
cp wn=(cp){cos(2.0*PI*flag/m),sin(2.0*PI*flag/m)};
for(int i=;i<n;i+=m)
{
cp w=(cp){,};int mid=m>>;
for0(j,mid-)
{
cp u=x[i+j],v=x[i+j+mid]*w;
x[i+j]=u+v;x[i+j+mid]=u-v;
w=w*wn;
}
}
}
if(flag==-)for0(i,n-)x[i].x/=n;
}
inline void work(int x)
{
//cout<<"XXXXX"<<' '<<x<<' '<<"XXXX"<<endl;
del[x]=;mx=;
for4(i,x)if(!del[y=e[i].go])
{
cnt=;
getdep(y,x,);get(mx);
for0(j,m-)a[j]=(cp){g[j],},b[j]=(cp){,};
for1(j,cnt)b[d[j]].x+=,g[d[j]]++;
//for0(j,m-1)cout<<j<<' '<<a[j].x<<' '<<b[j].x<<endl;
fft(a,m,);fft(b,m,);
for0(j,m-)c[j]=a[j]*b[j];
fft(c,m,-);
for0(j,m-)ret[j]+=c[j].x+0.5;
//for0(j,m-1)cout<<j<<' '<<c[j].x<<' '<<c[j].y<<endl;
}
for1(i,mx)ret[i]+=g[i],g[i]=;
for4(i,x)if(!del[y=e[i].go])
{
sum=s[y];rt=;
getrt(y,);
work(rt);
}
} int main() { freopen("input.txt","r",stdin); freopen("output.txt","w",stdout); n=read();
for1(i,n-)insert(read()+,read()+);
sum=n;f[rt=]=inf;
getrt(,);
work(rt);
double ans=0.0;
for1(i,n)
ans+=(double)ret[i]/(double)(i+);//cout<<i<<' '<<ret[i]<<endl;
printf("%.4f\n",n+*ans); return ; }

3451: Tyvj1953 Normal

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 125  Solved: 60
[Submit][Status]

Description

某天WJMZBMR学习了一个神奇的算法:树的点分治!
这个算法的核心是这样的:
消耗时间=0
Solve(树 a)
 消耗时间 += a 的 大小
 如果 a 中 只有 1 个点
  退出
 否则在a中选一个点x,在a中删除点x
 那么a变成了几个小一点的树,对每个小树递归调用Solve
我们注意到的这个算法的时间复杂度跟选择的点x是密切相关的。
如果x是树的重心,那么时间复杂度就是O(nlogn)
但是由于WJMZBMR比较傻逼,他决定随机在a中选择一个点作为x!
Sevenkplus告诉他这样做的最坏复杂度是O(n^2)
但是WJMZBMR就是不信>_<。。。
于是Sevenkplus花了几分钟写了一个程序证明了这一点。。。你也试试看吧^_^
现在给你一颗树,你能告诉WJMZBMR他的傻逼算法需要的期望消耗时间吗?(消耗时间按在Solve里面的那个为标准)

Input

第一行一个整数n,表示树的大小
接下来n-1行每行两个数a,b,表示a和b之间有一条边
注意点是从0开始标号的

Output

一行一个浮点数表示答案
四舍五入到小数点后4位
如果害怕精度跪建议用long double或者extended

Sample Input

3
0 1
1 2

Sample Output

5.6667

HINT

n<=30000

Source

BZOJ3451: Tyvj1953 Normal的更多相关文章

  1. BZOJ3451 Tyvj1953 Normal 点分治 多项式 FFT

    原文链接https://www.cnblogs.com/zhouzhendong/p/BZOJ3451.html 题目传送门 - BZOJ3451 题意 给定一棵有 $n$ 个节点的树,在树上随机点分 ...

  2. [BZOJ3451][Tyvj1953]Normal(点分治+FFT)

    https://www.cnblogs.com/GXZlegend/p/8611948.html #include<cmath> #include<cstdio> #inclu ...

  3. BZOJ3451 Tyvj1953 Normal 【期望 + 点分治 + NTT】

    题目链接 BZOJ3451 题解 考虑每个点产生的贡献,即为该点在点分树中的深度期望值 由于期望的线性,最后的答案就是每个点贡献之和 对于点对\((i,j)\),考虑\(j\)成为\(i\)祖先的概率 ...

  4. 【BZOJ3451】Tyvj1953 Normal 点分治+FFT+期望

    [BZOJ3451]Tyvj1953 Normal Description 某天WJMZBMR学习了一个神奇的算法:树的点分治!这个算法的核心是这样的:消耗时间=0Solve(树 a) 消耗时间 += ...

  5. bzoj 3451: Tyvj1953 Normal [fft 点分治 期望]

    3451: Tyvj1953 Normal 题意: N 个点的树,点分治时等概率地随机选点,代价为当前连通块的顶点数量,求代价的期望值 百年难遇的点分治一遍AC!!! 今天又去翻了一下<具体数学 ...

  6. 【BZOJ3451】Normal

    [BZOJ3451]Normal Description 某天WJMZBMR学习了一个神奇的算法:树的点分治! 这个算法的核心是这样的: 消耗时间=0 Solve(树 a) 消耗时间 += a 的 大 ...

  7. 【BZOJ3451】Normal (点分治)

    [BZOJ3451]Normal (点分治) 题面 BZOJ 题解 显然考虑每个点的贡献.但是发现似乎怎么算都不好计算其在点分树上的深度. 那么考虑一下这个点在点分树中每一次被计算的情况,显然就是其在 ...

  8. 【bzoj3451】Tyvj1953 Normal 期望+树的点分治+FFT

    题目描述 给你一棵 $n$ 个点的树,对这棵树进行随机点分治,每次随机一个点作为分治中心.定义消耗时间为每层分治的子树大小之和,求消耗时间的期望. 输入 第一行一个整数n,表示树的大小接下来n-1行每 ...

  9. BZOJ3451:Tyvj1953 Normal

    根据期望的线性性,答案就是 \(\sum\) 每个连通块出现次数的期望 而每个连通块次数的期望就是 \(\sum\) 连通块的根与每个点连通次数的期望 也就是对于一条路径 \((i,j)\),设 \( ...

随机推荐

  1. 对vector等STL标准容器的排序操作

    [+] STL提供的Sort 算法 所有sort算法介绍 sort 中的比较函数 sort 的稳定性 全排序 局部排序 nth_element 指定元素排序 partition 和stable_par ...

  2. 1072. Gas Station (30)

    先要求出各个加油站 最短的 与任意一房屋之间的 距离D,再在这些加油站中选出最长的D的加油站 ,该加油站 为 最优选项 (坑爹啊!).如果相同D相同 则 选离各个房屋平均距离小的,如果还是 相同,则 ...

  3. 实现c++的string的split功能

    今天写程序,遇到了一个要实现string.split()这个的一个函数.python里面有,qt里面有,c++里面没有.照着网上抄了一个,放在这里.有需要的时候直接拽过去用,否则老是写了小例子就扔,用 ...

  4. 【学习总结】Info.plist和pch文件的作用

      Info.plist   建立一个工程后,会在Supporting files文件夹下看到一个“Info.plist”的文件,该文件对工程做一些运行期的配置,非常重要,不能删除 项目中其他Plis ...

  5. owa Your request can't be completed right now. Please try again later.

    Your request can't be completed right now. Please try again later.

  6. js获取当前浏览器页面高度及宽度信息的方法

    var scrollLeft = Math.max(document.documentElement.scrollLeft, document.body.scrollLeft); var scroll ...

  7. 集成“支付宝” -b

    大致步骤 1.与支付宝签约获取相关参数 合作者身份 ID 与安全校验码 key2.下载需要导入的文件,做相应设置3.在自己的项目中集成支付的方法代码 详细步骤 1.获取合作者身份 ID 与安全校验码 ...

  8. collectionView代码创建

    @interface ViewController ()<UICollectionViewDataSource,UICollectionViewDelegateFlowLayout> @p ...

  9. (转载)shell日志分析常用命令

    shell日志分析常用命令总结 时间:2016-03-09 15:55:29来源:网络 导读:shell日志分析的常用命令,用于日志分析的shell脚本,统计日志中百度蜘蛛的抓取量.抓取最多的页面.抓 ...

  10. 100 doors

    Question There are 100 doors in a row that are all initially closed. You make 100 passes by the door ...