题意:有N个城市,M条街道(连接两点的距离),每条街道是单向的,现在要你设计多条路线覆盖所有的点,每条路线都是一个环,并且每个点仅能被一条路线覆盖且只经过一次(终始点除外),
分析:因为是有向圈,所以每个点的入度和出度应该都是1,故将一个点拆成两个点,入度点和出度点,然后用最佳匹配即可!(因为最佳匹配是求最大值,故我们把边权设为负值即可!)
注意:这题有重边,题目太不道德了,有重边都不说,还要猜的啊!有些题没说有重边就没重边,有些题没说有重边但是它就是有重边!无敌了都!
KM_match :
#include <iostream> #include <stdio.h> #include <string.h> using namespace std; const int inf=10000000; const int N=205; int m[N][N]; //邻接矩阵 int lx[N],ly[N]; //顶点标号 int fx[N],fy[N];//是否已经搜索过,S为寻找从i出发的增广轨时访问的x中的点的集合,T为访问的y中的点的集合。 int match[N]; int n; int dfs(int k) //从x[k]寻找增广路 { fx[k]=1; for(int i=1; i<=n; i++) { if(!fy[i]&& lx[k]+ly[i]==m[k][i])//相等子图 { fy[i]=1; if(match[i]==-1||dfs(match[i])) { match[i]=k; return 1; } } } return 0; } int KM_match() //求解最小权匹配 { int Min; for(int i=1; i<=n; i++) { lx[i]=-1;//x中顶点i的编号为与i关联的Y中边的最大权重 for(int j=1; j<=n; j++) if(lx[i]<m[i][j]) lx[i]=m[i][j];//初始化x的顶点标号 } memset(ly,0,sizeof(ly));//初始化y的顶点标号 memset(match,-1,sizeof(match)); for(int k=1; k<=n; k++) { while(1) { memset(fx,0,sizeof(fx)); memset(fy,0,sizeof(fy)); if(dfs(k)) break; Min=inf; for(int i=1; i<=n; i++) if(fx[i]) //x在交错树中 for(int j=1; j<=n; j++) if(!fy[j])//y不在交错树中,扩大子图 Min=min(Min,lx[i]+ly[j]-m[i][j]); //若匹配不成功,则修改顶点标号,找到d的值 for(int i = 1 ; i <= n ;i++) if(fx[i]) lx[i] -= Min ; for(int i = 1 ; i <= n ;i++) if(fy[i]) ly[i] += Min ; } } int sum=0; for(int i=1; i<=n; i++) sum+=m[match[i]][i]; return -sum; } int main() { int t; cin>>t; while(t--) { int M,ans; cin>>n>>M; for(int i=0; i<=n; i++) for(int j=0; j<=n; j++) m[i][j]=-inf;//即将边的权值取反 while(M--) { int u,v,w; cin>>u>>v>>w; if(m[u][v]<-w)//WA的地方,注意呀!!! m[u][v]=-w; } ans=KM_match(); printf("%d\n",ans); } return 0; }
最小费用最大流 :
也是拆点, (s,i,1,0) ,(i+n,t,1,0) ,(a,b+n,1,cost) ;
#include<cstdio> #include<cstring> #include<map> #include<vector> #include<cmath> #include<cstdlib> #include<stack> #include<queue> #include <iomanip> #include<iostream> #include<algorithm> using namespace std ; const int N=2500 ; const int inf=1<<30; struct node { int u,v,c,cost,next; } edge[N*100] ; int dist[N],pre[N],head[N],vist[N]; int g[N][N],top; void add(int u ,int v ,int c,int cost) { edge[top].u=u;edge[top].v=v;edge[top].c=c;edge[top].cost=cost;edge[top].next=head[u];head[u]=top++; edge[top].u=v;edge[top].v=u;edge[top].c=0;edge[top].cost=-cost;edge[top].next=head[v];head[v]=top++; } int SPFA(int s,int t,int n) { memset(pre,-1,sizeof(pre)); memset(vist,0,sizeof(0)); for(int i =0 ; i <= n ;i++) dist[i]=inf ; int i,u,v; vist[s]=1;dist[s]=0; queue<int>q ; q.push(s); while(!q.empty()) { u=q.front(); q.pop(); vist[u]=0; for( i = head[u] ; i!=-1 ; i=edge[i].next) { v=edge[i].v ; if(edge[i].c && dist[v] > dist[u]+edge[i].cost) { dist[v] = dist[u]+edge[i].cost ; pre[v]=i; if(!vist[v]) { vist[v]=1; q.push(v); } } } } if(dist[t]==inf) return 0; return 1; } int MFMC(int s,int t,int n) { int minflow,mincost=0,flow=0; while(SPFA(s,t,n)) { minflow=inf ; for(int i=pre[t];i!=-1;i=pre[edge[i].u]) minflow=min(minflow,edge[i].c) ; for(int i=pre[t];i!=-1;i=pre[edge[i].u]) { edge[i].c -= minflow; edge[i^1].c += minflow ; } mincost += dist[t]*minflow ; flow+=minflow ; } return mincost ; } int main() { int T,n,m,s,t,u,v,c; scanf("%d",&T); while(T--) { scanf("%d%d",&n,&m); top=0; memset(head,-1,sizeof(head)); s=0,t=2*n+1 ; while(m--) { scanf("%d%d%d",&u,&v,&c); add(u,v+n,1,c); } for(int i = 1 ; i <= n ; i++) add(s,i,1,0),add(i+n,t,1,0); int ans=MFMC(s,t,t+1); printf("%d\n",ans); } return 0; }