日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

P5631-最小mex生成树【线段树,并查集】

發(fā)布時間:2023/12/3 编程问答 25 豆豆
生活随笔 收集整理的這篇文章主要介紹了 P5631-最小mex生成树【线段树,并查集】 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

正題

題目鏈接:https://www.luogu.com.cn/problem/P5631


題目大意

nnn個點mmm條邊的一張圖,求mexmexmex值最小的一棵生成樹。


解題思路

考慮比較暴力的做法,枚舉答案,然后判斷其他邊能否構(gòu)成一棵生成樹。

發(fā)現(xiàn)一條邊會被重復加入多次,可以考慮不刪除其他不動的邊。

具體方法在線段樹上,對于邊權(quán)為www的邊,每次把www丟到[1,w?1]∪[w+1,∞][1,w-1]\cup[w+1,\infty][1,w?1][w+1,]這個區(qū)間。

然后在線段樹上往下走,維護一個不壓縮路徑但是按秩合并的可撤銷堆,每次走到一個節(jié)點就把這個節(jié)點上的邊加入,離開的時候就撤銷掉。這樣到葉節(jié)點時判斷是否是一棵生成樹就好了。

時間復雜度O(nlog?nlog?w)O(n\log n\log w)O(nlognlogw)


code

#include<cstdio> #include<cstring> #include<algorithm> #include<vector> #include<cctype> using namespace std; const int N=1e6+10,L=1e5+2; struct line{int x,y; }cl[N]; int n,m,cnt,fa[N],siz[N]; vector<line> v[N<<1]; int read(){int x=0,f=1;char c=getchar();while(!isdigit(c)){if(c=='-')f=-f;c=getchar();}while(isdigit(c)){x=(x<<1)+(x<<3)+c-'0';c=getchar();}return x*f; } int find(int x) {return (fa[x]==x)?x:find(fa[x]);} void unionm(int x,int y){x=find(x);y=find(y);if(x==y)return;if(siz[x]<siz[y])swap(x,y);cl[++cnt]=(line){x,y};siz[x]+=siz[y];fa[y]=x;return; } void clear_to(int w){while(cnt>w){int x=cl[cnt].x,y=cl[cnt].y;siz[x]-=siz[y];fa[y]=y;cnt--;}return; } void Change(int x,int L,int R,int l,int r,line w){if(L==l&&R==r){v[x].push_back(w);return;}int mid=(L+R)>>1;if(r<=mid)Change(x*2,L,mid,l,r,w);else if(l>mid)Change(x*2+1,mid+1,R,l,r,w);else Change(x*2,L,mid,l,mid,w),Change(x*2+1,mid+1,R,mid+1,r,w);return; } int Solve(int x,int l,int r){int top=cnt;for(int i=0;i<v[x].size();i++)unionm(v[x][i].x,v[x][i].y);if(l==r){if(cnt==n-1)return l;clear_to(top);return 0;}int mid=(l+r)>>1,k;if(k=Solve(x*2,l,mid))return k;if(k=Solve(x*2+1,mid+1,r))return k;clear_to(top);return 0; } int main() {n=read();m=read();for(int i=1;i<=n;i++)fa[i]=i,siz[i]=1;for(int i=1;i<=m;i++){int x=read(),y=read(),w=read()+1;if(w!=1)Change(1,1,L,1,w-1,(line){x,y});if(w!=L)Change(1,1,L,w+1,L,(line){x,y});}printf("%d\n",Solve(1,1,L)-1);return 0; }

總結(jié)

以上是生活随笔為你收集整理的P5631-最小mex生成树【线段树,并查集】的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。