2095: [Poi2010]Bridges

Time Limit: 10 Sec  Memory Limit: 259 MB

Description

YYD为了减肥,他来到了瘦海,这是一个巨大的海,海中有n个小岛,小岛之间有m座桥连接,两个小岛之间不会有两座桥,并且从一个小岛可以到另外任意一个小岛。现在YYD想骑单车从小岛1出发,骑过每一座桥,到达每一个小岛,然后回到小岛1。霸中同学为了让YYD减肥成功,召唤了大风,由于是海上,风变得十分大,经过每一座桥都有不可避免的风阻碍YYD,YYD十分ddt,于是用泡芙贿赂了你,希望你能帮他找出一条承受的最大风力最小的路线。

Input

输入:第一行为两个用空格隔开的整数n(2<=n<=1000),m(1<=m<=2000),接下来读入m行由空格隔开的4个整数a,b(1<=a,b<=n,a<>b),c,d(1<=c,d<=1000),表示第i+1行第i座桥连接小岛a和b,从a到b承受的风力为c,从b到a承受的风力为d。

Output

输出:如果无法完成减肥计划,则输出NIE,否则第一行输出承受风力的最大值(要使它最小)

Sample Input

4 4
1 2 2 4
2 3 3 4
3 4 4 4
4 1 5 4

Sample Output

4

HINT

注意:通过桥为欧拉回路
 
题解:注意到“最大值最小”这一关键词,不难转化为想到二分答案判合法性问题。
那么我们接下来要判断的就是“混合图是否存在欧拉回路”这一问题。
我们考虑先给无向边规定一个方向,但是在这种定义下,得到的图未必是一个欧拉图,即有的点入度大于出度,有的点出度大于入度。
接下来我们考虑给已经定向的无向边“反向”。
设i点入度与出度的差值为delta[i],那么对于每个点,delta[i]显然一定是偶数,因为连着它的一条边反向就会造成±2的改变;
那么我们要做到工作就是“反转”某些边,使得delta全为0为了实现目的,我们:
从源点向入度大于出度的点连流量为入度减出度/2的边,从入度小于出度向汇点的点连流量为出度减入度/2的边;
如果这样的边能够跑满,那么这个点就得到了完全的调整。
对一条无向边,连这条边的方向反向,流量为1的边,表示将这条边反向,两个点的入度与出度得到调整;
对这个网络求最大流就调整了尽可能多的无向边,源点和汇点所连边的流量都跑满时,
所有需要调整的边都被调整,出现了欧拉回路。
所以我们一开始记录一下与源点相连的边的流量和sum,再跑一边dinic/ISAP看是否满流即可。
代码实现:
 #include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int N=,M=,inf=0x7fffffff;
struct edge{int zhong,next,flow;};
int a[M],b[M],c[M],d[M],delta[N],n,sum,m;
struct NetWork_Flow
{
edge s[M<<];
int S,T,e,adj[N],cur[N];
int dist[N],hd,tl,q[N],cnt[N];
inline void add(int qi,int zhong,int flow)
{s[++e].zhong=zhong,s[e].next=adj[qi],adj[qi]=e,s[e].flow=flow;}
inline void bfs()
{
memset(cnt,,sizeof(cnt)),memset(dist,,sizeof(dist));
hd=,tl=,dist[T]=,q[++tl]=T;
register int i,x;
while(hd<=tl)
for(x=q[hd++],++cnt[dist[x]],i=adj[x];i;i=s[i].next)
if(s[i^].flow&&!dist[s[i].zhong])
dist[s[i].zhong]=dist[x]+,q[++tl]=s[i].zhong;
}
inline int Shoot(int rt,int maxf)
{
if(rt==T||!maxf)return maxf;
register int i,x,u,f,ret=;
for(i=cur[rt];i;i=s[i].next)
if(dist[s[i].zhong]+==dist[rt])
{
f=Shoot(s[i].zhong,min(maxf,s[i].flow));
ret+=f,maxf-=f,s[i].flow-=f,s[i^].flow+=f;
if(!maxf)return ret;
}
if(!(--cnt[dist[rt]]))dist[S]=T+;
++cnt[++dist[rt]],cur[rt]=adj[rt];
return ret;
}
inline int ISAP()
{
register int i;
memcpy(cur,adj,sizeof(adj));
int maxf=;bfs();
while(dist[S]<=T+)maxf+=Shoot(S,inf);
return maxf;
}
inline void build(int val)
{
register int i;
e=,sum=,memset(adj,,sizeof(adj));
memset(delta,,sizeof(delta));
for(i=;i<=m;++i)
{
if(c[i]<=val)--delta[a[i]],++delta[b[i]];
if(d[i]<=val)
add(b[i],a[i],),add(a[i],b[i],);
}
for(i=;i<=n;++i)
if(delta[i]>)sum+=delta[i]/,add(S,i,delta[i]/),add(i,S,);
else add(i,T,-delta[i]/),add(T,i,);
}
inline bool check(int val)
{
register int i;
build(val);
for(i=;i<=n;++i)
if(delta[i]&)return ;
return ISAP()==sum;
}
}G;
int main()
{
scanf("%d%d",&n,&m);
G.S=n+,G.T=n+;
register int i,j;
int l=,r=,mi,ans=inf;
for(i=;i<=m;++i)
{
scanf("%d%d%d%d",&a[i],&b[i],&c[i],&d[i]);
if(c[i]>d[i])swap(c[i],d[i]),swap(a[i],b[i]);
l=min(l,c[i]),r=max(r,d[i]);
}
while(l<=r)
{
mi=l+r>>;
if(G.check(mi))r=mi-,ans=mi;
else l=mi+;
}
if(ans==inf)puts("NIE");
else printf("%d\n",ans);
}

[BZOJ2095][Poi2010]Bridges 最大流(混合图欧拉回路)的更多相关文章

  1. BZOJ2095 POI2010 Bridges 【二分+混合图欧拉回路】

    BZOJ2095 POI2010 Bridges Description YYD为了减肥,他来到了瘦海,这是一个巨大的海,海中有n个小岛,小岛之间有m座桥连接,两个小岛之间不会有两座桥,并且从一个小岛 ...

  2. 【BZOJ-2095】Bridge 最大流 + 混合图欧拉回路 + 二分

    2095: [Poi2010]Bridges Time Limit: 10 Sec  Memory Limit: 259 MBSubmit: 604  Solved: 218[Submit][Stat ...

  3. bzoj2095: [Poi2010]Bridges(二分+混合图求欧拉回路)

    传送门 这篇题解讲的真吼->这里 首先我们可以二分一个答案,然后把所有权值小于这个答案的都加入图中 那么问题就转化为一张混合图(既有有向边又有无向边)中是否存在欧拉回路 首先 无向图存在欧拉回路 ...

  4. bzoj 2095: [Poi2010]Bridges(二分法+混合图的欧拉回路)

    [题意] 给定n点m边的无向图,对于边u,v,从u到v边权为c,从v到u的边权为d,问能够经过每条边一次且仅一次,且最大权值最小的欧拉回路. [思路] 二分答案mid,然后切断权值大于mid的边,原图 ...

  5. BZOJ2095:[POI2010]Bridges(最大流,欧拉图)

    Description YYD为了减肥,他来到了瘦海,这是一个巨大的海,海中有n个小岛,小岛之间有m座桥连接,两个小岛之间不会有两座桥,并且从一个小岛可以到另外任意一个小岛.现在YYD想骑单车从小岛1 ...

  6. BZOJ.2095.[POI2010]Bridges(最大流ISAP 二分 欧拉回路)

    题目链接 最小化最大的一条边,二分答案.然后就变成了给一张无向图定向使其为欧拉回路 二分答案后对于一个位置的两条边可能都保留,即双向边,需要给它定向:可能只保留小的一条,即单向边,不需考虑 如何给它定 ...

  7. POJ 1637 - Sightseeing tour - [最大流解决混合图欧拉回路]

    嗯,这是我上一篇文章说的那本宝典的第二题,我只想说,真TM是本宝典……做的我又痛苦又激动……(我感觉ACM的日常尽在这张表情中了) 题目链接:http://poj.org/problem?id=163 ...

  8. POJ 1637 Sightseeing tour(混合图欧拉回路+最大流)

    http://poj.org/problem?id=1637 题意:给出n个点和m条边,这些边有些是单向边,有些是双向边,判断是否能构成欧拉回路. 思路: 构成有向图欧拉回路的要求是入度=出度,无向图 ...

  9. BZOJ2095 [Poi2010]Bridges

    首先二分答案...然后这张图变成了有一些有向边,有一些无向边 然后就是混合图欧拉回路的判断 我们知道如果是有向图,它存在欧拉回路的等价条件是所有点的出度等于入度 对于混合图...先不管有向边,把无向边 ...

随机推荐

  1. virtualBox linux操作系统centos 挂载光盘

    virtualBox虚拟机linux centos 挂载光盘 第一步: 放入光盘 第二步: 挂载光驱 (/dev/cdrom) 1) 创建挂载点 mkdir /mnt/media 2) 挂载 moun ...

  2. Shader做剪影效果

    某渣渣甩了一个需求给我,并且说我不会写.我明知是激将法,但是想想这需求也太简单了,我好像也不怎么会QAQ.为了表示我对shader的热爱,写就写. 需求是这样的: 这是一个漂亮的MM,但是渣渣不想让人 ...

  3. 苹果针对on sale 的APP发的问题邮件

    2017年3月8日 上午8:07 发件人 Apple Dear Developer, Your app, extension, and/or linked framework appears to c ...

  4. Daily Scrum3 11.5

    昨天的任务已经完成,但是大家分析后发现进度稍有些慢.今天各自都在调整进度,不再拖延别人的工作. 今日任务: 杨伊:做问卷调查,准备用户体验篇内容. 徐钧鸿:把Xueba中Utility 向闸瓦移植 张 ...

  5. Daily Scrum (2015/10/25)

    今天终于到了周末的尾声,我们的组员也应该正常得投入到工作中了.这天晚上我(符美潇)和PM(潘礼鹏)和两个DEV开了一个小会,讨论一下我们本周的代码编写工作.我们了解到大家的代码阅读工作和相关知识的学习 ...

  6. Scurm Meeting 11.2

    成员 今日任务 明日计划 用时 徐越 写功能规格说明书,代码移植 创建数据库,代码移植 3h 赵庶宏 编写功能规格说明书,学习访问数据库代码,代码迁移 代码迁移 5h 武鑫 设计界面:独立完成一些简单 ...

  7. No.111_第四次团队会议

    后端的偏执 啊,这次又轮到我写团队博客了. 此时又是深夜,窗外漫天繁星.舍友的呼噜声惊吓了月亮,它害羞地跑回了云里去. 我关上灯拔掉机械,悄悄拿着电脑上了床,写这次的团队博客.曾经觉得自己绝对不会晚睡 ...

  8. 奔跑吧DKY——团队Scrum冲刺阶段-Day 6

    今日完成任务 谭鑫:制作相应动画人物,并实现人物动画 黄宇塘:制作相应动画人物,并实现人物动画,制作背景图 赵晓海:制作相应动画人物,并实现人物动画 方艺雯:制作相应动画人物,并实现人物动画,编写博客 ...

  9. 安卓端通过http对Mysql进行增删改查

    各类it学习视频,大家都可以看看哦!我自己本人都是通过这些来学习it只知识的! 下面是视频链接转自:http://www.cnblogs.com/yzxk/p/4749440.html Android ...

  10. 2018软工实践—Beta冲刺(2)

    队名 火箭少男100 组长博客 林燊大哥 作业博客 Beta 冲鸭鸭! 成员冲刺阶段情况 林燊(组长) 过去两天完成了哪些任务 协调组内工作 修改前端界面 展示GitHub当日代码/文档签入记录(组内 ...