zoukankan      html  css  js  c++  java
  • 【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 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(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
    ",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 }
  • 相关阅读:
    ORACLE日期时间函数大全
    Oracle 11.2.0.4 Data Guard 部署练习【一主一备/单机】
    Oracle 11.2 RAC 添加节点
    Oracle 11.2 RAC 删除节点
    Oracle 11.2 RAC on Redhat 6.5 安装最佳实践
    MongoDB复制集概念架构浅析
    Linux下MongoDB单实例的安装和配置详解
    抓包
    openstack安全组
    chrome headless 截图
  • 原文地址:https://www.cnblogs.com/chty/p/5924262.html
Copyright © 2011-2022 走看看