码迷,mamicode.com
首页 > 其他好文 > 详细

[CODECHEF]PRIMEDST

时间:2018-09-07 11:14:45      阅读:221      评论:0      收藏:0      [点我收藏+]

标签:algo   break   统计   swap   prime   fft   关于   pre   就是   

题意:求在树上任选一条路径,长度为质数的概率

直接点分,不难,之所以写是因为想记一下关于时间复杂度的东西...

我们采用按顺序加子树的方式统计答案,记一个深度有多少个点,卷积统计答案

如果当前节点的子树最深深度分别为$d_{1\cdots m}$,那么我们合并$i$时的数据规模为$\max\{d_{1\cdots i-1}\}+d_i$,如果直接做可以被卡到$n^2$

所以我们把$d_i$从小到大排序,这样做的总数据规模就是$\sum d_i$级别的,也就是不超过当前子树大小,符合点分的要求

#include<stdio.h>
#include<math.h>
#include<string.h>
#include<algorithm>
using namespace std;
typedef long long ll;
typedef double du;
const int T=50000,inf=2147483647;
const du pi=3.141592653589793238462643383;
int pr[T+10];
bool np[T+10];
void sieve(){
	int i,j,M=0;
	np[1]=1;
	for(i=2;i<=T;i++){
		if(!np[i])pr[++M]=i;
		for(j=1;j<=M&&i*pr[j]<=T;j++){
			np[i*pr[j]]=1;
			if(i%pr[j]==0)break;
		}
	}
}
struct comp{
	du x,y;
	comp(du a=0,du b=0){x=a;y=b;}
};
comp operator+(comp a,comp b){return comp(a.x+b.x,a.y+b.y);}
comp operator-(comp a,comp b){return comp(a.x-b.x,a.y-b.y);}
comp operator*(comp a,comp b){return comp(a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x);}
int rev[65536],N;
comp w[16][65536];
void pre(int n){
	int i,j,k;
	for(N=1,k=0;N<=n;N<<=1)k++;
	for(i=0;i<N;i++)rev[i]=(rev[i>>1]>>1)|((i&1)<<(k-1));
	for(i=2,k=0;i<=N;i<<=1,k++){
		for(j=0;j<i>>1;j++)w[k][j]=comp(cos(j*pi/(i/2)),sin(j*pi/(i/2)));
	}
}
void fft(comp*a,int on){
	int i,j,k,f;
	comp t;
	for(i=0;i<N;i++){
		if(i<rev[i])swap(a[i],a[rev[i]]);
	}
	for(i=2,f=0;i<=N;i<<=1,f++){
		for(j=0;j<N;j+=i){
			for(k=0;k<i>>1;k++){
				t=w[f][k];
				t.y*=on;
				t=t*a[i/2+j+k];
				a[i/2+j+k]=a[j+k]-t;
				a[j+k]=a[j+k]+t;
			}
		}
	}
	if(on==-1){
		for(i=0;i<N;i++)a[i].x/=N;
	}
}
int h[50010],nex[100010],to[100010],M;
void add(int a,int b){
	M++;
	to[M]=b;
	nex[M]=h[a];
	h[a]=M;
}
bool v[50010];
int siz[50010];
void dfs1(int fa,int x){
	siz[x]=1;
	for(int i=h[x];i;i=nex[i]){
		if(!v[to[i]]&&to[i]!=fa){
			dfs1(x,to[i]);
			siz[x]+=siz[to[i]];
		}
	}
}
int mn,cn,n;
void dfs2(int fa,int x){
	int i,mx=0;
	for(i=h[x];i;i=nex[i]){
		if(!v[to[i]]&&to[i]!=fa){
			dfs2(x,to[i]);
			mx=max(mx,siz[to[i]]);
		}
	}
	mx=max(mx,n-siz[x]);
	if(mx<mn){
		mn=mx;
		cn=x;
	}
}
int md[50010];
void dfs3(int fa,int x){
	md[x]=1;
	for(int i=h[x];i;i=nex[i]){
		if(!v[to[i]]&&to[i]!=fa){
			dfs3(x,to[i]);
			md[x]=max(md[x],md[to[i]]+1);
		}
	}
}
int p[50010],d[50010],td[50010];
void dfs4(int fa,int x,int d){
	td[d]++;
	for(int i=h[x];i;i=nex[i]){
		if(!v[to[i]]&&to[i]!=fa)dfs4(x,to[i],d+1);
	}
}
bool cmp(int x,int y){return md[x]<md[y];}
ll ans;
comp A[65536],B[65536];
void solve(int x){
	int i,j,M;
	dfs1(0,x);
	n=siz[x];
	mn=inf;
	dfs2(0,x);
	x=cn;
	v[x]=1;
	dfs3(0,x);
	M=0;
	for(i=h[x];i;i=nex[i]){
		if(!v[to[i]])p[++M]=to[i];
	}
	if(M==0)return;
	sort(p+1,p+M+1,cmp);
	memset(d,0,(md[p[M]]+1)<<2);
	d[0]=1;
	for(i=1;i<=M;i++){
		memset(td,0,(md[p[i]]+1)<<2);
		dfs4(x,p[i],1);
		pre(md[p[i-1]]+md[p[i]]);
		memset(A,0,N<<4);
		memset(B,0,N<<4);
		for(j=0;j<=md[p[i-1]];j++)A[j]=d[j];
		for(j=0;j<=md[p[i]];j++)B[j]=td[j];
		fft(A,1);
		fft(B,1);
		for(j=0;j<N;j++)A[j]=A[j]*B[j];
		fft(A,-1);
		for(j=1;j<=md[p[i-1]]+md[p[i]];j++){
			if(!np[j])ans+=llround(A[j].x);
		}
		for(j=0;j<=md[p[i]];j++)d[j]+=td[j];
	}
	for(i=h[x];i;i=nex[i]){
		if(!v[to[i]])solve(to[i]);
	}
}
int main(){
	sieve();
	int n,i,x,y;
	scanf("%d",&n);
	for(i=1;i<n;i++){
		scanf("%d%d",&x,&y);
		add(x,y);
		add(y,x);
	}
	solve(1);
	printf("%.7lf",ans/(double)((ll)n*(n-1)/2));
}

CC昨晚不给提交,今早又复活了...

[CODECHEF]PRIMEDST

标签:algo   break   统计   swap   prime   fft   关于   pre   就是   

原文地址:https://www.cnblogs.com/jefflyy/p/9602429.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!