Processing math: 100%

【UOJ#61】【UR#5】怎样更有力气

暴力

题目大意:

给定一棵树的草图,第 i 天给出 ui,vi,wi,你可以任选草图上位于 uivi 之间的两个点并实际修一条路,代价为 wi

p 条限制,第 i 条限制形如 ui,vi,ti,表示第 ti 天,uivi 之间不能修路。

求让 n 个点连通的最小代价。

题解:

题目就是要求最小生成树,所以容易想到按 w 排序以后分别考虑。

对于同一个连通块的点,我们可以将它们放在一起考虑。

我们考虑对一个 u,v,将它们所在的所有连通块找出来。然后若两个连通块在这条路径里的点不是两两有边,则将这两个连通块连起来。

由于最多有 p 条限制,枚举两个连通块,但不连边的次数只有 O(p) 次,是合理的。

同一连通块的点不一定在树上连续,而暴力跳树的复杂度无法接受。简单的方法是若一个结点和父亲在同一个连通块,就用并查集并起来。每次暴力往上跳。这样暴力往上跳而没有合并的次数也只有 O(p) 次。

并未仔细证明,感性理解得到的东西

然后枚举两个连通块并考虑连边即可。

Code:

  1. #include<iostream>
  2. #include<algorithm>
  3. #include<vector>
  4. #include<map>
  5. #include<set>
  6. using namespace std;
  7. const int N=3e5+5;
  8. typedef long long LL;
  9. LL ans;
  10. int n,m,p,head[N],cnt,nxt[N],dep[N],sz[N],son[N],fa[N],top[N];
  11. vector<pair<int,int> >vc[N];
  12. inline bool cmp(int a,int b){return dep[a]<dep[b];}
  13. struct dsu{
  14. int fa[N];
  15. inline int find(int x){while(x!=fa[x])x=fa[x]=fa[fa[x]];return x;}
  16. inline void merge(int x,int y){x=find(x),y=find(y);if(x>y)swap(x,y);if(x!=y)fa[y]=x;}
  17. inline void init(){for(int i=1;i<=n;++i)fa[i]=i;}
  18. }A,B;
  19. struct que{
  20. int u,v,w,t;
  21. inline bool operator<(const que&rhs)const{return w<rhs.w;}
  22. }d[N];
  23. void dfs(int now){
  24. sz[now]=1,son[now]=0;
  25. for(int to=head[now];to;to=nxt[to]){
  26. dep[to]=dep[now]+1;
  27. dfs(to);
  28. sz[now]+=sz[to];
  29. if(!son[now]||sz[son[now]]<sz[to])son[now]=to;
  30. }
  31. }
  32. void dfs2(int now){
  33. if(son[now])top[son[now]]=top[now],dfs2(son[now]);
  34. for(int to=head[now];to;to=nxt[to])if(to!=son[now])dfs2(top[to]=to);
  35. }
  36. inline int LCA(int x,int y){
  37. while(top[x]!=top[y])if(dep[top[x]]>dep[top[y]])x=fa[top[x]];else y=fa[top[y]];
  38. return dep[x]<dep[y]?x:y;
  39. }
  40. bool check(int u,int v,int t,int x){
  41. if(dep[x]<dep[t])return 0;
  42. return(LCA(u,x)==x||LCA(v,x)==x);
  43. }
  44. void work(int u,int v,int w,vector<pair<int,int> >&vc){
  45. static int sz[N],sta[N];
  46. static bool vis[N];
  47. static map<int,int>mp[N];
  48. int top=0;
  49. int t=LCA(u,v);
  50. for(int x=u;dep[x]>=dep[t];){
  51. int tp=B.find(x);
  52. if(dep[tp]<dep[t])tp=t;
  53. if(A.find(tp)==A.find(fa[tp]))B.merge(fa[tp],tp);
  54. int bel=A.find(tp);
  55. if(!vis[bel]){sta[++top]=bel,vis[bel]=1;}
  56. sz[bel]+=dep[x]-dep[tp]+1;
  57. x=fa[tp];
  58. }
  59. for(int x=v;dep[x]>=dep[t];){
  60. int tp=B.find(x);
  61. if(dep[tp]<dep[t])tp=t;
  62. if(A.find(tp)==A.find(fa[tp]))B.merge(fa[tp],tp);
  63. int bel=A.find(tp);
  64. if(!vis[bel]){sta[++top]=bel,vis[bel]=1;}
  65. sz[bel]+=dep[x]-dep[tp]+1;
  66. x=fa[tp];
  67. }
  68. --sz[A.find(t)];
  69. for(auto it:vc){
  70. int x=it.first,y=it.second;
  71. if(check(u,v,t,x)&&check(u,v,t,y)){
  72. int fx=A.find(x),fy=A.find(y);
  73. if(fx!=fy)++mp[fx][fy],++mp[fy][fx];
  74. }
  75. }
  76. static set<int>st;
  77. st.clear();
  78. int dd=0;
  79. for(int i=1;i<=top;++i)st.insert(sta[i]);
  80. for(int i=1;i<=top&&st.size();++i){
  81. static int D[N];int TOP=0;
  82. for(auto j:st)if(A.find(sta[i])!=A.find(j)&&(LL)sz[sta[i]]*sz[j]>mp[sta[i]][j]){
  83. D[++TOP]=j;
  84. ++dd;
  85. A.merge(sta[i],j);
  86. }
  87. for(int i=1;i<=TOP;++i)st.erase(D[i]);
  88. }
  89. for(int i=1;i<=top;++i)mp[sta[i]].clear(),sz[sta[i]]=0,vis[sta[i]]=0;
  90. ans+=(LL)w*dd;
  91. }
  92. int main(){
  93. ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
  94. cin>>n>>m>>p;
  95. for(int i=2;i<=n;++i){
  96. cin>>fa[i];
  97. nxt[i]=head[fa[i]],head[fa[i]]=i;
  98. }
  99. dfs(dep[1]=1),dfs2(top[1]=1);
  100. A.init(),B.init();
  101. for(int i=1;i<=m;++i){
  102. cin>>d[i].u>>d[i].v>>d[i].w;
  103. d[i].t=i;
  104. }
  105. sort(d+1,d+m+1);
  106. while(p--){
  107. int u,v,t;
  108. cin>>t>>u>>v;
  109. vc[t].emplace_back(u,v);
  110. }
  111. for(int i=1;i<=m;++i)
  112. work(d[i].u,d[i].v,d[i].w,vc[d[i].t]);
  113. cout<<ans<<'\n';
  114. return 0;
  115. }

Gitalking ...