刚学网络流的我这里有一道非常好的"网络流练手题"------[HNOI2006]超级英雄.

  记得很久以前真的有这个节目来着,还是大兵主持的.

  其实这是一道匈牙利板子大水题,但对于我们这种刚学网络流的 juruo 碰到什么题不用网络流做就非常的不爽,于是就有了这个 超级英雄 网络流+二分版.

  这里直接二分答案,每次将源点和所有的锦囊连边,容量为1;再将锦囊和对应的问题连边,容量为1;最后是将问题和汇点T连边,容量还是1.然后跑一波Dinic,看是否满流,这道"网络流练手题就搞定了".

  最后再吐槽一下:这道题的网络流打法非常非常优秀(比匈牙利慢而且代码量翻倍),没事干的时候可以打着玩儿......

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#define inf (1<<30)
#define maxn (300050)
#define maxm (300050)
#define ll long long
#define il inline
#define RG register
using namespace std;
il int gi(){
  RG int x=0,q=1; RG char ch=getchar();
  while( ( ch<'0' || ch>'9' ) && ch!='-' ) ch=getchar();
  if( ch=='-' ) q=-1,ch=getchar();
  while(ch>='0' && ch<='9') x=x*10+ch-48,ch=getchar();
  return q*x;
}

int S,T,n,m,num=1,F,que[maxn*5],level[maxn];
struct data{ int a,b; }ques[maxn];

int head[maxn],nxt[maxm],to[maxm],w[maxm];
il void add( int u,int v,int d ){
  nxt[++num]=head[u];to[num]=v;w[num]=d;head[u]=num;
  nxt[++num]=head[v];to[num]=u;w[num]=0;head[v]=num; }

il bool bfs(){
  for(int i=S;i<=T;i++) level[i]=0;
  que[0]=S,level[S]=1;
  RG int hd=0,tl=1;
  while(hd<tl){
    RG int x=que[hd++];
    if(x==T) return true;
    for(int i=head[x];i;i=nxt[i]){
      RG int v=to[i];
      if(w[i] && !level[v]){
        level[v]=level[x]+1;
        que[tl++]=v;
      }
    }
  }
  return false;
}

il int dfs(RG int x,RG int maxf){
  if(x==T) return maxf;
  RG int ans=0;
  for(RG int i=head[x];i;i=nxt[i]){
    RG int v=to[i],f=w[i];
    if(level[v]==level[x]+1 && f){
      RG int minn=min(f,maxf-ans);
      f=dfs(v,minn);
      w[i]-=f,w[i^1]+=f,ans+=f;
      if(ans==maxf) break;
    }
  }
  if(!ans) level[x]=0;
  return ans;
}

il void Dinic(){ while(bfs()) F+=dfs(S,inf); }

il bool check(int mid){
  memset(head,0,sizeof(head)); num=1; F=0;
  for(int i=1;i<=n;i++) add(S,i,1);
  for(int i=1;i<=mid;i++){
    add(ques[i].a,i+n,1); add(ques[i].b,i+n,1); add(i+n,T,1);
  }
  Dinic();
  if(F>=mid) return 1;
  else return 0;
}

il void init(){
  n=gi(),m=gi();S=0,T=n+m+1;
  for(int i=1;i<=m;i++) ques[i].a=gi()+1,ques[i].b=gi()+1;
}

il void work(){
  int l=0,r=m,ans=0;
  while(l<=r){
    RG int mid=(l+r)>>1;
    if( check(mid) ) l=mid+1,ans=mid;
    else r=mid-1;
  }
  printf("%d\n",ans);
}

int main(){ init(); work(); return 0; }

  

     最后顺便再附上匈牙利版代码:

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int map[1001][1001],club[1001];
int n,m;
bool vis[1001];
bool dfs(int x){
  for(int i=0;i<n;i++)
    if(!vis[i]&&map[x][i]){
      vis[i]=1;
      if( !club[i]||dfs(club[i]) ){ club[i]=x; return true; }
    }
  return false;
}

int main(){
  scanf("%d%d",&n,&m);
  for(int i=1;i<=m;i++){
    int x,y;
    scanf("%d%d",&x,&y);
    map[i][x]=map[i][y]=1;
  }
  int ans=0;
  for(int i=1;i<=m;i++){
    memset(vis,0,sizeof(vis));
    if(dfs(i))ans++;
    else break;
  }
  printf("%d\n",ans);
  return 0;
}

  

[HNOI2006]超级英雄 网络流+二分版的更多相关文章

  1. [HNOI2006]超级英雄(二分+网络流)

    [HNOI2006]超级英雄 题目描述 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金.主持人问题准备了若干道题目, ...

  2. BZOJ 1191: [HNOI2006]超级英雄Hero 二分匹配

    1191: [HNOI2006]超级英雄Hero Description 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或 ...

  3. BZOJ 1191: [HNOI2006]超级英雄Hero(二分图匹配)

    云神说他二分图匹配从来都是用网络流水过去的...我要发扬他的精神.. 这道题明显是二分图匹配.网络流的话可以二分答案+最大流.虽然跑得很慢.... -------------------------- ...

  4. bzoj 1191: [HNOI2006]超级英雄Hero

    1191: [HNOI2006]超级英雄Hero Time Limit: 10 Sec  Memory Limit: 162 MB 二分图匹配... Description 现在电视台有一种节目叫做超 ...

  5. bzoj 1191: [HNOI2006]超级英雄Hero 并查集 || 匈牙利算法

    1191: [HNOI2006]超级英雄Hero Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1804  Solved: 850[Submit][S ...

  6. bzoj 1191 [HNOI2006]超级英雄Hero(最大基数匹配)

    1191: [HNOI2006]超级英雄Hero Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2813  Solved: 1331[Submit][ ...

  7. POJ Secret Milking Machine 【网络流+二分】

    题意:各一个地图,两点之间有若干条路,要在节点1和节点n之间行走t次(就是问1到n的路径数至少为t,每一条路径不能有重复),问所有路径里面最长的部分(这个题目特别强调,不是路径长度和,是路径中相邻两点 ...

  8. 1191: [HNOI2006]超级英雄Hero

    1191: [HNOI2006]超级英雄Hero Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1996  Solved: 946[Submit][S ...

  9. [luogu P2319] [HNOI2006]超级英雄

    [luogu P2319] [HNOI2006]超级英雄 题目描述 现在电视台有一种节目叫做超级英雄,大概的流程就是每位选手到台上回答主持人的几个问题,然后根据回答问题的多少获得不同数目的奖品或奖金. ...

随机推荐

  1. 工程师倾情奉献-Win7 ISO 精简操作说明

    1.前提条件 a)本文档内容只适用于32bit win7 install ISO,其它OS不能保证兼容 b)示范文件为win7-ultimate-rtm-32-en-us-rdvd.iso 2.准备待 ...

  2. read命令读取用户输入

    read命令用于从终端或文件中读取用户输入,它读取整行输入,如果没有指定名称,读取的行被赋值给内部变量REPLY.read命令常用选项:-a,-p,-s,-t,-n 1.REPLY变量 $readhe ...

  3. Linux命令 文件压缩及压缩命令

    gzip [功能说明] 文件的压缩 #gizp属于GNU软件,总性能不错,是Linux系统首选的压缩工具,tar归档命令的-z参数也是利用gzip/gunzip来解压缩 [语法格式] Gip[选项][ ...

  4. MongoDB--架构搭建(主从、副本集)之主从

    此章节讲述主从架构 主从架构  -- 目前已经不建议使用,推荐使用复制集 主从配置可以在配置文件中配置 从节点可以在启动之后使用命令追加主节点,db.source.insert({"host ...

  5. Python:Anaconda安装虚拟环境到指定路径

    1 曾经的困扰 有段时间,想使用基于不同python版本的anaconda,就直接从官网下载了两个不同的anaconda版本进行安装.刚开始的时候,还觉得也没啥问题.用了一小段时间,在安装其他的第三方 ...

  6. flask笔记二

    web表单 web表单是浏览者和网之间的一个互动平台,完成浏览器和服务器之间的数据交互. 1.用Flask-WTF来处理表单 (1)在根目录下编辑扩展配置--config.py CSRF_ENABLE ...

  7. PHP设计模式:工厂方法

    示例代码详见https://github.com/52fhy/design_patterns 工厂方法 工厂方法是针对每一种产品提供一个工厂类.通过不同的工厂实例来创建不同的产品实例. 相比简单工厂, ...

  8. 谈谈Nancy中让人又爱又恨的Diagnostics【上篇】

    前言 在Nancy中有个十分不错的功能-Diagnostics,可以说这个功能让人又爱又恨. 或许我们都做过下面这样的一些尝试: 记录某一个功能用到的相关技术信息 记录下网站的访问记录 全局配置某些框 ...

  9. jquery删除表格行

    $(".mingxirmspan").click(function(){ $(this).closest("tr").remove(); })

  10. php中有关合并某一字段键值相同的数组合并

    <?php function combine($array,$start,$key,$newkey){ static $new; //静态变量 foreach($array as $k=> ...