国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

PrayerOJ1823: 每條邊的最小生成樹

2019-11-11 05:01:08
字體:
來源:轉載
供稿:網友

http://PRayer.hustoj.com/problem.php?id=1823 CF上好像也有題 這題其實蠻巧妙的 顯然暴力的代碼也難搞 那我直接說標算了; 吧原圖的最小生成樹搞出來 對于詢問 如果邊不在我們求出來的最小生成樹上,那一定會形成一個環 我們只要把環里除詢問邊外最長的邊刪掉就可以了 妥妥的; 但是證明去找到最長邊呢 我第一個反應就是暴力; 后來zyy大佬說用lca 對啊 如果我們把詢問邊砍掉 就是破環 lac其兩個端點,必然會訪問其環上的各個邊 所以在搞倍增表時,隨手搞一個倍增max表就好了

#include<iostream>#include<cstdio>#include<cstdlib>#include<cmath>#include<cstring>#include<algorithm>#include<cstring>#include<string>#define Ll long longusing namespace std;struct cs{ int x,y,z,num;}a[10001],aa[10001];//讀入a,aa備份 int g[1001][1001],deep[1001],bz[1001][15],ma[1001][15];//g來存最小生成樹,ma就是倍增max bool b[10001];//表示邊i在不在原來的最小生成樹里 int father[1001];//我用kruskalint n,m,xx,yy,ans,start;bool cmp(cs a,cs b){ return a.z<b.z;}int getfa(int x){ if(father[x]==x)return x; father[x]=getfa(father[x]); return father[x];}void dfs(int x,int y,int z){ deep[x]=z; bz[x][0]=y; ma[x][0]=g[x][y]; for(int i=1;i<=n;i++) if(g[i][x]>0&&i!=y)dfs(i,x,z+1);}void bzb(){ for(int j=1;(1<<j)<=n;j++) for(int i=1;i<=n;i++){ bz[i][j]=bz[bz[i][j-1]][j-1]; ma[i][j]=max(ma[i][j-1],ma[bz[i][j-1]][j-1]); }}void happytogether(int x,int y){ if(x==y)return; while(1){ int j=0; if(bz[x][j]==bz[y][j]){ xx=max(xx,ma[y][j]); xx=max(xx,ma[x][j]); return; } while(bz[x][j]!=bz[y][j])j++; j--; xx=max(xx,ma[y][j]); xx=max(xx,ma[x][j]); x=bz[x][j]; y=bz[y][j]; }}int upone(int stdd,int x){ while(deep[x]!=stdd){ int j=0; while(deep[bz[x][j]]>=stdd)j++; xx=max(xx,ma[x][j-1]); x=bz[x][j-1]; } return x;}void lca(int x,int y){ if(deep[x]>deep[y])swap(x,y); y=upone(deep[x],y); happytogether(x,y);} main(){ scanf("%d%d",&n,&m); for(int i=1;i<=m;i++){ scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].z); a[i].num=i; aa[i].x=a[i].x;aa[i].y=a[i].y;aa[i].z=a[i].z; } sort(a+1,a+m+1,cmp); for(int i=1;i<=n;i++)father[i]=i; for(int i=1;i<=m;i++){ xx=getfa(a[i].x); yy=getfa(a[i].y); if(xx==yy)continue; ans+=a[i].z; father[xx]=yy; b[a[i].num]=1; g[a[i].y][a[i].x]=g[a[i].x][a[i].y]=a[i].z; start=a[i].x; } dfs(start,0,1); bzb(); for(int i=1;i<=m;i++) if(b[i])cout<<ans<<endl;else{ xx=0; lca(aa[i].x,aa[i].y); cout<<ans-xx+aa[i].z<<endl; }}
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 北票市| 黑水县| 越西县| 建平县| 石楼县| 琼中| 孝昌县| 太仆寺旗| 江津市| 宁远县| 洛隆县| 正定县| 固原市| 溆浦县| 通渭县| 金溪县| 肃宁县| 广水市| 汨罗市| 灵石县| 剑川县| 军事| 繁峙县| 梅州市| 宁乡县| 张北县| 承德县| 福海县| 克拉玛依市| 沅陵县| 通许县| 黑山县| 武平县| 虹口区| 唐山市| 安平县| 连南| 平谷区| 台南县| 府谷县| 蒙山县|