先上题目 bzoj3669: [Noi2014]魔法森林

这道题首先每一条边都有一个a,b 我们按a从小到大排序 每次将一条路劲入队 当然这道题权在边上 所以我们将边化为点去连接他的两个端点

当然某两个点我用的是并查集维护 其实也可以在树上直接查询 但是这样比较方便 同时我们维护某个点极其子树的最大值所在的位置 每入一条边 如果两个端点已经联通就找出权值的边删掉之后连上新的边顺便更新答案 如果没联通就直接连上边就好 这样就解决问题了

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int M=,inf=0x3f3f3f3f;
int read(){
int ans=,f=,c=getchar();
while(c<''||c>''){if(c=='-') f=-; c=getchar();}
while(c>=''&&c<=''){ans=ans*+(c-''); c=getchar();}
return ans*f;
}
int n,m,ans=inf;
int c[M][],fa[M],p[M];
int mx[M],v[M],rev[M];
bool isrt(int x){return c[fa[x]][]!=x&&c[fa[x]][]!=x;}
struct node{int a,b,u,v;}e[M];
bool cmp(node a,node b){return a.a<b.a;}
int find(int x){return p[x]==x?x:p[x]=find(p[x]);}
void up(int x){
int l=c[x][],r=c[x][];
mx[x]=x;
if(v[mx[l]]>v[mx[x]]) mx[x]=mx[l];
if(v[mx[r]]>v[mx[x]]) mx[x]=mx[r];
}
void down(int x){
int l=c[x][],r=c[x][];
if(rev[x]){
rev[x]=; rev[l]^=; rev[r]^=;
swap(c[x][],c[x][]);
}
}
void rotate(int x){
int y=fa[x],z=fa[y],l=,r=;
if(c[y][]==x) l=,r=;
if(!isrt(y)) c[z][c[z][]==y]=x;
fa[y]=x; fa[x]=z; fa[c[x][r]]=y;
c[y][l]=c[x][r]; c[x][r]=y;
up(y); up(x);
}
int st[M],top;
void splay(int x){
st[++top]=x; for(int i=x;!isrt(i);i=fa[i]) st[++top]=fa[i];
while(top) down(st[top--]);
while(!isrt(x)){
int y=fa[x],z=fa[y];
if(!isrt(y)){
if(c[z][]==y^c[y][]==x) rotate(x);
else rotate(y);
}
rotate(x);
}
}
void acs(int x0){
for(int x=x0,y=;x;splay(x),c[x][]=y,up(x),y=x,x=fa[x]);
splay(x0);
}
void mrt(int x){acs(x); rev[x]^=;}
void link(int x,int y){mrt(x); fa[x]=y;}
void cut(int x,int y){mrt(x); acs(y); c[y][]=fa[x]=; up(y);}
int push_max(int x,int y){mrt(x); acs(y); return mx[y];}
int main()
{
n=read(); m=read();
for(int i=;i<=n;i++) p[i]=i;
for(int i=;i<=m;i++) e[i].u=read(),e[i].v=read(),e[i].a=read(),e[i].b=read();
sort(e+,e++m,cmp);
for(int i=;i<=m;i++){
int from=e[i].u,to=e[i].v,a=e[i].a,b=e[i].b;
if(find(from)==find(to)){
int now=push_max(from,to);
if(v[now]>b){cut(now,e[now-n].v); cut(now,e[now-n].u);}
else{
if(find()==find(n)) ans=min(ans,a+v[push_max(,n)]);
continue;
}
}
else p[find(from)]=find(to);
v[i+n]=b; mx[i+n]=i+n;
link(from,i+n); link(to,i+n);
if(find()==find(n)) ans=min(ans,a+v[push_max(,n)]);
}
if(ans==inf) printf("-1\n");
else printf("%d\n",ans);
return ;
}

最新文章

  1. [Hadoop in Action] 第7章 细则手册
  2. awk使用shell变量
  3. 非常棒的Android对话框效果
  4. UIView的setNeedsDisplay和setNeedsLayout
  5. iOS,一行代码进行RSA、DES 、AES、MD5加密、解密
  6. vs2012中VC连接mysql
  7. 需要弥补的那部分SQL
  8. jQuery中的尺寸及位置的取和设
  9. 22、TTS技术
  10. Win7 64bit 成功安装ArcView3.X
  11. svn 树冲突
  12. Android广播——短信拦截
  13. mysql中变量character_set_connection的具体作用
  14. 【GO】关于GO的浅显总结
  15. [IR] Concept Search and LDA
  16. 添加FTP用户(vsftpd)
  17. layui选项卡同步问题
  18. Mycat&#160;中间件配置初探与入门操作
  19. python-ConfigParser模块【读写配置文件】
  20. InitializingBean和DisposableBean

热门文章

  1. Hackerrank - [Algo] Matrix Rotation
  2. Viewer.js 图片预览插件使用
  3. 树莓派搭建 Hexo 博客(一)
  4. windows下git hub的GUI软件配置与使用
  5. 团队项目-第五次Scrum 会议
  6. MapReduce 并行编程理论基础
  7. valgrind使用
  8. 【python】Python3中出现&#39;gbk&#39; codec can&#39;t encode characte的成功解决方法?
  9. PAT 甲级 1007 Maximum Subsequence Sum
  10. 关于php网络爬虫phpspider