首页 > 代码库 > 【NOIP2008】传纸条
【NOIP2008】传纸条
【描述】 Description
小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了。幸运的是,他们可以通过传纸条来进行交流。纸条要经由许多同学传到对方手里,小渊坐在矩阵的左上角,坐标(1,1),小轩坐在矩阵的右下角,坐标(m,n)。从小渊传到小轩的纸条只可以向下或者向右传递,从小轩传给小渊的纸条只可以向上或者向左传递。
在活动进行中,小渊希望给小轩传递一张纸条,同时希望小轩给他回复。班里每个同学都可以帮他们传递,但只会帮他们一次,也就是说如果此人在小渊递给小轩纸条的时候帮忙,那么在小轩递给小渊的时候就不会再帮忙。反之亦然。
还有一件事情需要注意,全班每个同学愿意帮忙的好感度有高有低(注意:小渊和小轩的好心程度没有定义,输入时用0表示),可以用一个0-100的自然数来表示,数越大表示越好心。小渊和小轩希望尽可能找好心程度高的同学来帮忙传纸条,即找到来回两条传递路径,使得这两条路径上同学的好心程度只和最大。现在,请你帮助小渊和小轩找到这样的两条路径。
【题解】
用这道题来练习费用流,还是拆点的思想。
1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<cstdlib> 5 #include<ctime> 6 #include<cmath> 7 #include<algorithm> 8 using namespace std; 9 #define MAXN 1001010 #define INF 100000000011 struct node{int y,next,v,w,rel;}e[MAXN*5];12 int n,m,len,S,T,ans,Link[MAXN],q[MAXN*10],vis[MAXN],dis[MAXN],lastedge[MAXN],lastnode[MAXN];13 inline int read()14 {15 int x=0,f=1; char ch=getchar();16 while(!isdigit(ch)) {if(ch==‘-‘) f=-1; ch=getchar();}17 while(isdigit(ch)) {x=x*10+ch-‘0‘; ch=getchar();}18 return x*f;19 }20 void insert(int x,int y,int v,int w)21 {22 e[++len].next=Link[x];Link[x]=len;e[len].y=y;e[len].v=v;e[len].w=-w;e[len].rel=len+1;23 e[++len].next=Link[y];Link[y]=len;e[len].y=x;e[len].v=0;e[len].w=w;e[len].rel=len-1;24 }25 void build()26 {27 n=read(); m=read();28 for(int i=1;i<=n;i++)29 for(int j=1;j<=m;j++)30 {31 int x=(i-1)*m+j,y=x*2,v=read(); x=y-1;32 insert(x,y,1,v);33 if(j<m) insert(y,y+1,1,0);34 if(i<n) insert(y,y+2*m-1,1,0);35 }36 S=1; T=n*m*2-1;37 e[S].v=2;38 }39 bool spfa()40 {41 memset(vis,0,sizeof(vis));42 memset(dis,10,sizeof(dis));43 int head=0,tail=1,oo=vis[0];44 q[++tail]=S; vis[S]=1; dis[S]=0;45 while(++head<=tail)46 {47 int x=q[head],y;48 for(int i=Link[q[head]];i;i=e[i].next)49 {50 y=e[i].y;51 if(e[i].v&&dis[x]+e[i].w<dis[y])52 {53 dis[y]=dis[x]+e[i].w;54 if(!vis[y])55 {56 q[++tail]=y;57 vis[y]=1;58 }59 lastnode[y]=x; lastedge[y]=i;60 }61 }62 vis[x]=0;63 }64 return dis[T]<oo;65 }66 void cost()67 {68 int d=INF;69 for(int i=T;i!=S;i=lastnode[i])70 if(e[lastedge[i]].v<d)71 d=e[lastedge[i]].v;72 for(int i=T;i!=S;i=lastnode[i])73 {74 int now=lastedge[i];75 e[now].v-=d;76 e[e[now].rel].v+=d;77 ans+=d*(-e[now].w);78 }79 }80 void solve()81 {82 while(spfa())83 cost();84 printf("%d\n",ans);85 }86 int main()87 {88 //freopen("cin.in","r",stdin);89 //freopen("cout.out","w",stdout);90 build();91 solve();92 return 0;93 }
【NOIP2008】传纸条
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。