【NOIP2008】传纸条

【描述】 Description
小渊和小轩是好朋友也是同班同学,他们在一起总有谈不完的话题。一次素质拓展活动中,班上同学安排做成一个m行n列的矩阵,而小渊和小轩被安排在矩阵对角线的两端,因此,他们就无法直接交谈了。幸运的是,他们可以通过传纸条来进行交流。纸条要经由许多同学传到对方手里,小渊坐在矩阵的左上角,坐标(1,1),小轩坐在矩阵的右下角,坐标(m,n)。从小渊传到小轩的纸条只可以向下或者向右传递,从小轩传给小渊的纸条只可以向上或者向左传递。
在活动进行中,小渊希望给小轩传递一张纸条,同时希望小轩给他回复。班里每个同学都可以帮他们传递,但只会帮他们一次,也就是说如果此人在小渊递给小轩纸条的时候帮忙,那么在小轩递给小渊的时候就不会再帮忙。反之亦然。
还有一件事情需要注意,全班每个同学愿意帮忙的好感度有高有低(注意:小渊和小轩的好心程度没有定义,输入时用0表示),可以用一个0-100的自然数来表示,数越大表示越好心。小渊和小轩希望尽可能找好心程度高的同学来帮忙传纸条,即找到来回两条传递路径,使得这两条路径上同学的好心程度只和最大。现在,请你帮助小渊和小轩找到这样的两条路径。

 

【题解】

用这道题来练习费用流,还是拆点的思想。

 1 #include
 2 #include
 3 #include
 4 #include
 5 #include
 6 #include
 7 #include
 8 using namespace std;
 9 #define MAXN 10010
10 #define INF 1000000000
11 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(j1,1,0);
34             if(i2*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】传纸条)