zoukankan      html  css  js  c++  java
  • 【LA2531 训练指南】足球联赛 【最大流】

    题意:

      有n支队伍进行比赛,每支队伍需要打的比赛数目相同。每场比赛恰好一支队伍胜,另一支败。给出每支队伍目前胜的场数和败的场数,以及每两支队伍还剩下的比赛场数,确定所有可能的冠军的球队。(获胜场数最多的是冠军,可以并列)。

    分析

      一只队伍如果可能得冠军,那么就一定可以通过调整,未来的几场比赛的结果使自己赢得场次最多。否则便不可能成为冠军。

       在判断第i支队伍有无可能成为冠军时:首先,第i支队伍得对局i全部取得胜利,得到i胜利的总场数为total。然后判断其他队伍的对局,能否互相限制使得,任何队伍胜利的场数都不超过total。

      建模方法:

     对每两支队伍(u,v)构造一个X结点,从S引一条弧过来,容量为这两支队伍还需要比赛的场数。对每支队伍u构造一个Y结点,引一条弧到T,容量为total-win[u].然后每个(u,v)结点向u和v结点个连一条无穷大得弧。这样当且仅当,从S出发得每条弧都是满载得时候,当前队伍才有可能得冠军。

      

      1 #include <cstdio>
      2 #include <cstring>
      3 #include <iostream>
      4 #include <algorithm>
      5 #include <queue>
      6 #include <vector>
      7 
      8 using namespace std;
      9 const int maxn=1000+10;
     10 const int MAXN=30;
     11 const int maxm=30000+100;
     12 const int INF=2147000000;
     13 struct Dinic{
     14     int head[maxn],Next[maxm],to[maxm],cap[maxm],flow[maxm],from[maxm];
     15     int sz,n,m,s,t;
     16     bool vis[maxn];
     17     int cur[maxn],d[maxn];
     18     void init(int n){
     19         this->n=n;
     20         memset(head,-1,sizeof(head));
     21         this->sz=-1;
     22     }
     23     void add_edge(int a,int b,int c){
     24         ++sz;
     25         to[sz]=b;
     26         cap[sz]=c;flow[sz]=0;from[sz]=a;
     27         Next[sz]=head[a];head[a]=sz;
     28         ++sz;
     29         to[sz]=a;
     30         cap[sz]=c;flow[sz]=c;from[sz]=b;
     31         Next[sz]=head[b];head[b]=sz;
     32     }
     33     bool BFS(){
     34         memset(vis,0,sizeof(vis));
     35         queue<int>Q;
     36         vis[s]=1;
     37         d[s]=0;
     38         Q.push(s);
     39         while(!Q.empty()){
     40             int u=Q.front();Q.pop();
     41             for(int i=head[u];i!=-1;i=Next[i]){
     42                 int v=to[i];
     43                 if(!vis[v]&&cap[i]>flow[i]){
     44                     vis[v]=1;
     45                     d[v]=d[u]+1;
     46                     Q.push(v);
     47                 }
     48             }
     49         }
     50         return vis[t];
     51    }
     52     int DFS(int x,int a){
     53         if(x==t||a==0)return a;
     54         int Flow=0,f;
     55         for(int& i=cur[x];i!=-1;i=Next[i]){
     56             int v=to[i];
     57             if(d[v]==d[x]+1&&(f=DFS(v,min(a,cap[i]-flow[i])))>0){
     58                 Flow+=f;
     59                 flow[i]+=f;
     60                 flow[i^1]-=f;
     61                 a-=f;
     62                 if(a==0)break;
     63             }
     64         }
     65         return Flow;
     66     }
     67     int Maxflow(int s,int t){
     68         this->s=s,this->t=t;
     69         int Flow=0;
     70         while(BFS()){
     71             for(int i=0;i<=n;i++)
     72              cur[i]=head[i];
     73             Flow+=DFS(s,INF);
     74         }
     75         return Flow;
     76     }
     77 }dinic;
     78 int T,n;
     79 int win[MAXN],lose[MAXN];
     80 int G[MAXN][MAXN];
     81 vector<int>ans;
     82 bool judge(int num){
     83     int total=win[num];
     84     for(int i=1;i<=n;i++){
     85         total+=G[num][i];
     86     }
     87     //cout<<total<<endl;
     88 
     89     int allS=(n*n-3*n+2)/2;
     90     int NUM=0;
     91     dinic.init(allS+n+2);
     92     dinic.s=0,dinic.t=allS+n+1;
     93     for(int i=1;i<=n;i++){
     94             if(i==num)continue;
     95         for(int j=i+1;j<=n;j++){
     96          //   cout<<i<<" "<<j<<" "<<G[i][j]<<endl;
     97             if(j==num)continue;
     98             NUM++;
     99             dinic.add_edge(0,NUM,G[i][j]);
    100             dinic.add_edge(NUM,allS+i,INF);
    101             dinic.add_edge(NUM,allS+j,INF);
    102         }
    103     }
    104 
    105     for(int i=1;i<=n;i++){
    106         if(i==num)continue;
    107         if(total<win[i])return false;
    108         dinic.add_edge(allS+i,dinic.t,total-win[i]);
    109     }
    110     dinic.Maxflow(0,allS+n+1);
    111     for(int i=0;i<=dinic.sz;i+=2){
    112         if(dinic.from[i]==0){
    113             if(dinic.flow[i]<dinic.cap[i])
    114                 return false;
    115         }
    116     }
    117     return true;
    118 }
    119 int main(){
    120     scanf("%d",&T);
    121     for(int t=1;t<=T;t++){
    122        // if(t!=1)printf("
    ");
    123         ans.clear();
    124         scanf("%d",&n);
    125         for(int i=1;i<=n;i++){
    126             scanf("%d%d",&win[i],&lose[i]);
    127         }
    128         for(int i=1;i<=n;i++){
    129             for(int j=1;j<=n;j++){
    130                 scanf("%d",&G[i][j]);
    131             }
    132         }
    133        /* if(judge(1))
    134             printf("Yes 1");*/
    135 
    136         for(int i=1;i<=n;i++){
    137             if(judge(i))
    138                 ans.push_back(i);
    139         }
    140         sort(ans.begin(),ans.end());
    141         for(int i=0;i<ans.size();i++){
    142             if(i!=0)printf(" ");
    143             printf("%d",ans[i]);
    144         }
    145         printf("
    ");
    146     }
    147 return 0;
    148 }
    View Code
  • 相关阅读:
    nodeName,nodeValue未知 xml 入库方案 The ElementTree iterparse Function
    如何:执行大型 XML 文档的流式转换 大XML文件解析入库的一个方法
    python curl_get-pip.py Installing with get-pip.py
    iptables List the rules in a chain or all chains
    goroutine 分析 协程的调度和执行顺序 并发写 run in the same address space 内存地址 闭包 存在两种并发 确定性 非确定性的 Go 的协程和通道理所当然的支持确定性的并发方式(
    数据库业界
    The MEAN stack is a modern replacement for the LAMP (Linux, Apache, MySQL, PHP/Python) stack
    Using Groovy To Import XML Into MongoDB
    虚拟机网络模式 桥接 网桥 交换机
    防止有内存泄漏并尽可能快地释放所有内存是内存管理的重要组成部分 长时间运行进程的内存泄漏
  • 原文地址:https://www.cnblogs.com/LQLlulu/p/9304332.html
Copyright © 2011-2022 走看看