首页 > 代码库 > HDU 2813
HDU 2813
http://acm.hdu.edu.cn/showproblem.php?pid=2813
裸二分图最优匹配,需要用两个map把武将名字映射到点的序号上
#include <iostream>#include <cstdio>#include <cstring>#include <map>using namespace std;const int N=210;const int INF=0x3f3f3f3f;int nx,ny;int linker[N],lx[N],ly[N],slack[N]; int visx[N],visy[N],w[N][N];int DFS(int x){ visx[x]=1; for(int y=1;y<=ny;y++){ if(visy[y]) continue; int tmp=lx[x]+ly[y]-w[x][y]; if(tmp==0){ visy[y]=1; if(linker[y]==-1 || DFS(linker[y])){ linker[y]=x; return 1; } }else if(slack[y]>tmp){ slack[y]=tmp; } } return 0;}int KM(){ int i,j; memset(linker,-1,sizeof(linker)); memset(ly,0,sizeof(ly)); for(i=1;i<=nx;i++) for(j=1,lx[i]=-INF;j<=ny;j++) if(w[i][j]>lx[i]) lx[i]=w[i][j]; for(int x=1;x<=nx;x++){ for(i=1;i<=ny;i++) slack[i]=INF; while(1){ memset(visx,0,sizeof(visx)); memset(visy,0,sizeof(visy)); if(DFS(x)) break; int d=INF; for(i=1;i<=ny;i++) if(!visy[i] && d>slack[i]) d=slack[i]; for(i=1;i<=nx;i++) if(visx[i]) lx[i]-=d; for(i=1;i<=ny;i++) if(visy[i]) ly[i]+=d; else slack[i]-=d; } } int res=0; for(i=1;i<=ny;i++) if(linker[i]!=-1) res+=w[linker[i]][i]; return res;}int main(){ int n,m,k ; while(~scanf("%d%d%d",&n,&m,&k)) { nx=n;ny=m; for(int i=0 ;i<N ;i++) for(int j=0 ;j<N ;j++) w[i][j]=-INF ; int p1=1,p2=1 ; map <string,int> mp1,mp2 ; for(int i=0 ;i<k ;i++) { char a[205],b[205] ; int v ; scanf("%s%s%d",a,b,&v) ; string s1(a),s2(b) ; if(!mp1[s1])mp1[s1]=p1++ ; if(!mp2[s2])mp2[s2]=p2++ ; w[mp1[s1]][mp2[s2]]=-v ; } int ans=KM(); printf("%d\n",-ans); } return 0;}
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。