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

歡迎訪問 生活随笔!

生活随笔

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

编程问答

图论 —— 网络流 —— 最大流 —— Dinic 算法

發(fā)布時間:2025/3/17 编程问答 19 豆豆
生活随笔 收集整理的這篇文章主要介紹了 图论 —— 网络流 —— 最大流 —— Dinic 算法 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

【概述】

Dinic 算法在 EK 算法的基礎(chǔ)上進行了優(yōu)化,其時間復(fù)雜度為 O(n*n*m)。

Dinic 在找增廣路的時也是找最短增廣路, 但與 EK 算法不同的是 Dinic?算法并不是每次 bfs 只找一個增廣路,他會首先通過一次 bfs 為所有點添加一個標號,構(gòu)成一個層次圖,然后在層次圖中尋找增廣路進行更新。

【基本思想】

1.初始化容量網(wǎng)絡(luò)與網(wǎng)絡(luò)流

2.構(gòu)造殘量網(wǎng)絡(luò),根據(jù)殘留網(wǎng)絡(luò)通過 BFS 計算層次網(wǎng)絡(luò),若匯點不在層次圖中(匯點層次為 -1),則結(jié)束

3.在層次網(wǎng)絡(luò)中使用一次 DFS 進行增廣,DFS 執(zhí)行完畢,該階段的增廣也執(zhí)行完畢

4.轉(zhuǎn)步驟 2

【求最大流】

1.一般方式

使用鄰接矩陣的形式,適用于點較少的情況。

int n,m;//n個點m條邊 int S,T;//源點、匯點 int level[N];//存儲結(jié)點分層層數(shù) struct Edge{int cap;int flow; }edge[N][N]; bool bfs(){//構(gòu)造層次網(wǎng)絡(luò)memset(level,0,sizeof(level));queue<int> Q;Q.push(S);level[S]=1;while(!Q.empty()){int x=Q.front();Q.pop();for(int y=1;y<=n;y++){if(!level[y]&&edge[x][y].cap>edge[x][y].flow){level[y]=level[x]+1;Q.push(y);}}}return level[T]!=0; }int dfs(int x,int cp){//計算可增加流量if(x==n)return cp;int flow=cp;//記錄從x到t的最小殘量for(int y=1;y<=n;y++){if(level[x]+1==level[y]){if(edge[x][y].cap>edge[x][y].flow){int minn=min(flow,edge[x][y].cap-edge[x][y].flow);int newFlow=dfs(y,minn);edge[x][y].flow+=newFlow;edge[y][x].flow-=newFlow;flow-=newFlow;}}if(flow==0)break;}return cp-flow; } int dinic(){int flow=0;int tf=0;while(bfs()){while(tf=dfs(1,INF)){flow+=tf;}}return flow; } int main(){scanf("%d%d",&n,&m);memset(edge,0,sizeof(edge));while(m--){int x,y,w;scanf("%d%d%d",&x,&y,&w);edge[x][y].cap+=w;//便于處理重邊}S=1,T=n;printf("%d\n",dinic());return 0; }

2.vector 鄰接表實現(xiàn)

使用 vector 鄰接表與結(jié)構(gòu)體結(jié)合,適用于點較多的情況,但速度稍慢于鄰接矩陣。

struct Edge{int from,to;int cap,flow;Edge(){}Edge(int from,int to,int cap,int flow):from(from),to(to),cap(cap),flow(flow){}}; int n,m; //結(jié)點數(shù),邊數(shù)(含反向弧) int S,T; //源點、匯點 vector<Edge> edges; //邊表,edges[e]和edges[e^1]互為反向弧 vector<int> G[N]; //鄰接表,G[i][j]表示結(jié)點i的第j條邊在e數(shù)組中的序號 bool vis[N]; //BFS使用,標記一個節(jié)點是否被遍歷過 int dis[N]; //dis[i]表從起點s到i點的距離(層次) int cur[N]; //cur[i]表當前正訪問i節(jié)點的第cur[i]條弧 void addEdge(int from,int to,int cap){edges.push_back( Edge(from,to,cap,0) );edges.push_back( Edge(to,from,0,0) );int m=edges.size();G[from].push_back(m-2);G[to].push_back(m-1); } bool BFS(){//構(gòu)建層次網(wǎng)絡(luò)memset(vis,0,sizeof(vis));dis[S]=0;vis[S]=true;queue<int> Q;//用來保存節(jié)點編號Q.push(S);while(!Q.empty()){int x=Q.front();Q.pop();for(int y=0;y<G[x].size();y++){Edge& e=edges[G[x][y]];if(!vis[e.to] && e.cap>e.flow){vis[e.to]=true;dis[e.to]=dis[x]+1;Q.push(e.to);}}}return vis[T]; }int DFS(int x,int cp){//cp表示從s到x目前為止所有弧的最小殘量if(x==T || cp==0)return cp;int flow=0,newFlow;//flow用來記錄從x到t的最小殘量for(int &y=cur[x];y<G[x].size();y++){Edge &e=edges[G[x][y]];if(dis[x]+1==dis[e.to]){int minn=min(cp,e.cap-e.flow);newFlow=DFS(e.to,minn);if(newFlow>0){e.flow+=newFlow;edges[G[x][y]^1].flow-=newFlow;flow+=newFlow;cp-=newFlow;if(cp==0)break;}}}return flow; } int Dinic(){int flow=0;while(BFS()){memset(cur,0,sizeof(cur));flow+=DFS(S,INF);}return flow; }int main(){int n,m;scanf("%d%d",&n,&m);//節(jié)點編號從1到n,邊編號從0到m-1for(int i=1;i<=n;i++)G[i].clear();edges.clear();for(int i=0;i<m;i++){int u,v,w;scanf("%d%d%d",&u,&v,&w);addEdge(u,v,w);}S=1,T=n;printf("%d\n",Dinic());return 0; }

【求最小割】

使用鄰接數(shù)組與 STL 中的 set 容器,在求最大流的過程中,將最小割求出。

其原理是最大流最小割定理:網(wǎng)絡(luò)的最大流的流量等于最小割的容量

struct Edge{int from,to;int cap,flow;Edge(){}Edge(int from,int to,int cap,int flow):from(from),to(to),cap(cap),flow(flow){} }; int n,m; //結(jié)點數(shù),邊數(shù)(含反向弧) int S,T; //源點、匯點 vector<Edge> edges; //邊表,edges[e]和edges[e^1]互為反向弧 vector<int> G[N]; //鄰接表,G[i][j]表示結(jié)點i的第j條邊在e數(shù)組中的序號 bool vis[N]; //BFS使用,標記一個節(jié)點是否被遍歷過 int dis[N]; //dis[i]表從起點s到i點的距離(層次) int cur[N]; //cur[i]表當前正訪問i節(jié)點的第cur[i]條弧 set<int> cutSet; //最小割 bool flag; //是否求最小割 void addEdge(int from,int to,int cap){edges.push_back( Edge(from,to,cap,0) );edges.push_back( Edge(to,from,0,0) );int m=edges.size();G[from].push_back(m-2);G[to].push_back(m-1); } bool BFS(){//構(gòu)建層次網(wǎng)絡(luò)memset(vis,0,sizeof(vis));dis[S]=0;vis[S]=true;//將超級源點加入最小割if(flag)cutSet.insert(S);queue<int> Q;//用來保存節(jié)點編號Q.push(S);while(!Q.empty()){int x=Q.front();Q.pop();for(int y=0;y<G[x].size();y++){Edge& e=edges[G[x][y]];if(!vis[e.to] && e.cap>e.flow){vis[e.to]=true;dis[e.to]=dis[x]+1;Q.push(e.to);if(flag)//記錄最小割元素cutSet.insert(e.to);}}}return vis[T]; }int DFS(int x,int cp){//cp表示從s到x目前為止所有弧的最小殘量if(x==T || cp==0)return cp;int flow=0,newFlow;//flow用來記錄從x到t的最小殘量for(int &y=cur[x];y<G[x].size();y++){Edge &e=edges[G[x][y]];if(dis[x]+1==dis[e.to]){int minn=min(cp,e.cap-e.flow);newFlow=DFS(e.to,minn);if(newFlow>0){e.flow+=newFlow;edges[G[x][y]^1].flow-=newFlow;flow+=newFlow;cp-=newFlow;if(cp==0)break;}}}return flow; } int Dinic(){int flow=0;while(BFS()){memset(cur,0,sizeof(cur));flow+=DFS(S,INF);}return flow; } int main(){scanf("%d%d",&n,&m);while(m--){int x,y,w;scanf("%d%d%d",&x,&y,&w);addEdge(x,y,w);}S=1,T=n;flag=false;//不統(tǒng)計最小割所屬元素int minCut=Dinic();//計算最小割的值printf("The Min-Cut and Max-Flow:%d\n",minCut);flag=true;//統(tǒng)計最小割所屬元素BFS();//構(gòu)建層次網(wǎng)絡(luò)printf("The Number of Min Cut:%d\n",cutSet.size());//最小割個數(shù)printf("The Min Cut Set:");set<int>::iterator it=cutSet.begin();for(;it!=cutSet.end();it++)//最小割元素printf("%d ",*it);printf("\n");return 0; }

總結(jié)

以上是生活随笔為你收集整理的图论 —— 网络流 —— 最大流 —— Dinic 算法的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

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

主站蜘蛛池模板: 国产三级三级三级 | 日本黄色片段 | 精品视频91 | 福利在线免费 | 欧美丝袜一区二区三区 | 男男play呻吟动漫网站 | 国产亚洲欧美精品久久久www | 揄拍成人国产精品视频 | 天堂影视av| 超薄肉色丝袜一二三 | 免费看美女被靠到爽的视频 | 丁香七月激情 | 中文字幕素人 | 亚洲国产成人精品女人久久 | 日本性爱视频在线观看 | 欧美精品久久久 | 中文字幕一级 | 韩国三级hd中文字幕的背景音乐 | 成人高潮片免费网站 | 日本va欧美va国产激情 | 中文字幕一区二区三区人妻 | 亚洲福利视频网 | 午夜视频1000 | 国产高清av在线 | 成人免费在线视频 | 黑丝久久 | 免费看一级黄色片 | 99热只有| 99久久婷婷国产综合精品 | 日韩国产欧美一区二区三区 | 精品人伦一区二区三区蜜桃免费 | 国产卡一卡二卡三 | 一本一道波多野结衣av黑人 | 日韩三级成人 | 亚洲春色在线 | 国产特级av | 国产精久久久 | 日本女优在线看 | 98精品视频 | 国产色宗合| 久伊人| av老司机在线 | 亚洲妇熟xx妇色黄蜜桃 | 久久丝袜美腿 | 亚洲视频欧洲视频 | 国产精品毛片久久久 | 制服丝袜一区在线 | 北岛玲在线 | 欧美激情福利 | jlzzjizz在线播放观看 | 大毛片| 邻居少妇张开双腿让我爽一夜 | 日本a视频| 亚洲一线在线观看 | 亚洲妇女av | 男生操女生网站 | 玉丸(双性调教) | 黑人添美女bbb添高潮了 | 四虎精品在线 | 一级淫片在线观看 | 一级成人免费 | 日韩综合精品 | www麻豆视频 | a天堂亚洲 | 在哪里可以看黄色片 | 日韩福利一区二区 | 999久久 | 日本一级片在线观看 | 这里精品 | 亚洲综合视频在线播放 | 亚洲人成人| 亚洲一区二区三区成人 | 91原创视频在线观看 | 孕妇毛片| 久久在草 | 国产精品三级久久久久久电影 | a级淫片| 黄色日本视频 | 免费啪啪网 | 欧美乱妇一区二区三区 | 在线日韩免费 | 国产夫妻久久 | a级黄色片| 男女ss视频 | 999精品视频在线观看 | 可以免费观看av的网站 | 亚洲成熟女性毛茸茸 | 亚洲日本中文字幕在线 | 亚洲国产97| 视频一区二区三区精品 | 99久热在线精品996热是什么 | 国产精品扒开腿做爽爽 | 国产精品欧美一区喷水 | 香蕉色网 | 亚洲乱码一区二区 | 国产第一页在线播放 | 91干网 | 色日韩 | av综合导航 |