【BZOJ1468】Tree [点分治]
Tree
Time Limit: 10 Sec Memory Limit: 64 MB[Submit][Status][Discuss]
Description
  给你一棵TREE,以及这棵树上边的距离,问有多少对点它们两者间的距离小于等于K。
Input
  第一行一个n,接下来n-1行边描述管道,按照题目中写的输入,接下来是一个k。
Output
  仅包括一个整数,表示有多少对点之间的距离小于等于k。
Sample Input
  7
1 6 13
6 3 9
3 5 7
4 1 3
2 4 20
4 7 2
10
1 6 13
6 3 9
3 5 7
4 1 3
2 4 20
4 7 2
10
Sample Output
  5
HINT
  n<=40000
Solution
  树上路径统计问题,运用点分。
  每次处理与重心相关的路径,发现如果直接处理两点之间比较困难,我们想到了将所有点加入一个数组,用指针判断加起来<=K的个数,这样的话不一定全是简单路径,但是我们只要减去每个子树中这样操作的条数就一定只剩下简单路径了。
  点分大概的步骤:
  1.找出重心;
  2.计算经过该重心的路径相关需要求的;
  3.去掉重心对于每棵子树继续做以上过程。
Code
 
1 #include<iostream> 2 #include<algorithm> 3 #include<cstdio> 4 #include<cstring> 5 #include<cstdlib> 6 #include<cmath> 7 using namespace std; 8 9 const int ONE=80001; 10 11 int n,K; 12 int x,y,z; 13 int next[ONE],first[ONE],go[ONE],w[ONE],tot; 14 int center_vis[ONE]; 15 int Max,dist[ONE],num; 16 int d[ONE]; 17 int center,Ans; 18 19 int get() 20 { 21 int res,Q=1; char c; 22 while( (c=getchar())<48 || c>57) 23 if(c=='-')Q=-1; 24 if(Q) res=c-48; 25 while((c=getchar())>=48 && c<=57) 26 res=res*10+c-48; 27 return res*Q; 28 } 29 30 int Add(int u,int v,int z) 31 { 32 next[++tot]=first[u]; first[u]=tot; go[tot]=v; w[tot]=z; 33 next[++tot]=first[v]; first[v]=tot; go[tot]=u; w[tot]=z; 34 } 35 36 namespace PointF 37 { 38 struct power 39 { 40 int maxx,size; 41 }S[ONE]; 42 43 void Getsize(int u,int father) 44 { 45 S[u].size=1; 46 S[u].maxx=0; 47 for(int e=first[u];e;e=next[e]) 48 { 49 int v=go[e]; 50 if(v==father || center_vis[v]) continue; 51 Getsize(v,u); 52 S[u].size+=S[v].size; 53 S[u].maxx=max(S[u].maxx,S[v].size); 54 } 55 } 56 57 void Getcenter(int u,int father,int total) 58 { 59 S[u].maxx=max(S[u].maxx,total-S[u].size); 60 if(Max>S[u].maxx) 61 { 62 Max=S[u].maxx; 63 center=u; 64 } 65 66 for(int e=first[u];e;e=next[e]) 67 { 68 int v=go[e]; 69 if(v==father || center_vis[v]) continue; 70 Getcenter(v,u,total); 71 } 72 } 73 74 void Getdist(int u,int father,int value) 75 { 76 dist[++num]=value; 77 for(int e=first[u];e;e=next[e]) 78 { 79 int v=go[e]; 80 if(v==father || center_vis[v]) continue; 81 Getdist(v,u,value+w[e]); 82 } 83 } 84 85 int Calc(int u,int value) 86 { 87 int res=0; 88 num=0; 89 90 Getdist(u,0,value); 91 sort(dist+1,dist+num+1); 92 93 int l=1,r=num; 94 95 while(l<r) 96 { 97 while(dist[l]+dist[r]>K && l<r) r--; 98 res+=r-l; 99 l++; 100 } 101 return res; 102 } 103 104 void Dfs(int u) 105 { 106 Max=n; 107 Getsize(u,0); 108 109 center=u; 110 Getcenter(u,0,S[u].size); 111 center_vis[center]=true; 112 113 Ans+=Calc(center,0); 114 for(int e=first[center];e;e=next[e]) 115 { 116 int v=go[e]; 117 if(center_vis[v]) continue; 118 Ans-=Calc(v,w[e]); 119 Dfs(v); 120 } 121 } 122 } 123 124 125 126 int main() 127 { 128 n=get(); 129 for(int i=1;i<n;i++) 130 { 131 x=get(); y=get(); z=get(); 132 Add(x,y,z); 133 } 134 135 K=get(); 136 PointF::Dfs(1); 137 138 printf("%d",Ans); 139 }

 
 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号