题意

\(n\)个人\(m\)场比赛\((1 \le n \le 10000, 0 \le m \le 10000)\),给出每场比赛的两个选手,求赢得最多的人最少赢的场数。

分析

二分最多人赢的场数,那么我们就得到了所有人赢的场次的上界。所以我们可以考虑网络流模型。

题解

对于二分的值\(d\),我们建\(m\)个点表示比赛,\(n\)个点表示人。对于每场比赛\(i\),连边\((s, i)\),上界为\(1\),连边\((i, u_i), (i, v_i)\),上界为\(1\)。对于每个人\(i\),连边\((i, t)\),上界为\(d\)。然后判断最大流是否是\(m\)即可。

然而这题卡isap,所以用dinic来做到单次最大流\(O((m+n)\sqrt{n})\)。(然而isap用一些奇葩的减枝也能过。

#include <bits/stdc++.h>
using namespace std;
inline int getint() {
int x=0;
char c=getchar();
for(; c<'0'||c>'9'; c=getchar());
for(; c>='0'&&c<='9'; x=x*10+c-'0', c=getchar());
return x;
}
const int vN=20015, N=10015, oo=~0u>>1;
int ihead[vN], cnt=1, win[N], X[N], Y[N];
struct E {
int next, to, cap;
}e[vN<<3];
void add(int x, int y) {
e[++cnt]=(E){ihead[x], y, 0}; ihead[x]=cnt;
e[++cnt]=(E){ihead[y], x, 0}; ihead[y]=cnt;
}
int isap(int s, int t, int n) {
static int gap[vN], cur[vN], d[vN], p[vN];
memset(d, 0, sizeof(int)*(n+1));
memset(cur, 0, sizeof(int)*(n+1));
memset(gap, 0, sizeof(int)*(n+1));
gap[0]=n;
int r=0;
for(int x=s, f, i; d[s]<n && d[s]<100;) {
for(i=cur[x]; i && !(e[i].cap && d[x]==d[e[i].to]+1); i=e[i].next);
if(i) {
p[e[i].to]=cur[x]=i;
if((x=e[i].to)==t) {
for(f=oo, x=t; x!=s; f=min(f, e[p[x]].cap), x=e[p[x]^1].to);
for(r+=f, x=t; x!=s; e[p[x]].cap-=f, e[p[x]^1].cap+=f, x=e[p[x]^1].to);
}
}
else {
if(!--gap[d[x]]) {
break;
}
d[x]=n;
for(i=ihead[x]; i; i=e[i].next) {
if(e[i].cap && d[x]>d[e[i].to]+1) {
d[x]=d[e[i].to]+1;
cur[x]=i;
}
}
++gap[d[x]];
if(x!=s) {
x=e[p[x]^1].to;
}
}
}
return r;
}
int n, m, s, t;
bool check(int mid) {
static int rest[N], match[N];
fill(rest+1, rest+1+n, mid);
int need=0;
for(int i=1; i<=m; ++i) {
int x=X[i], y=Y[i];
if(rest[x]<rest[y]) {
swap(x, y);
}
if(!rest[x]) {
++need;
match[i]=0;
}
else {
match[i]=x;
--rest[x];
}
}
if(!need) {
return 1;
}
int ct=1;
for(int i=1; i<=m; ++i) {
if(!match[i]) {
e[++ct].cap=1, e[++ct].cap=0;
e[++ct].cap=1, e[++ct].cap=0;
e[++ct].cap=1, e[++ct].cap=0;
}
else {
e[++ct].cap=0, e[++ct].cap=1;
if(X[i]==match[i]) {
e[++ct].cap=0, e[++ct].cap=1;
e[++ct].cap=1, e[++ct].cap=0;
}
else {
e[++ct].cap=1, e[++ct].cap=0;
e[++ct].cap=0, e[++ct].cap=1;
}
}
}
for(int i=1; i<=n; ++i) {
e[++ct].cap=rest[i], e[++ct].cap=mid-rest[i];
}
return isap(s, t, t)==need;
}
int main() {
n=getint(), m=getint(), s=m+n+1, t=s+1;
for(int i=1; i<=m; ++i) {
int x=X[i]=getint(), y=Y[i]=getint();
add(s, i);
add(i, m+x);
add(i, m+y);
if(win[x]>win[y]) {
swap(x, y);
}
++win[x];
}
int l=0, r=0;
for(int i=1; i<=n; ++i) {
add(m+i, t);
r=max(r, win[i]);
}
while(l<=r) {
int mid=(l+r)>>1;
check(mid)?(r=mid-1):(l=mid+1);
}
printf("%d\n", r+1);
return 0;
}

【BZOJ】1532: [POI2005]Kos-Dicing的更多相关文章

  1. 【BZOJ】1537: [POI2005]Aut- The Bus

    [算法]DP+线段树求区间max(二维偏序) [题解] 状态转移方程:f[i]=max(f[j]+v[i]),x[j]<x[i]&&y[j]<y[i]. 观察j的条件限制显 ...

  2. 【BZOJ】1529 [POI2005]ska Piggy banks

    [算法](强连通分量)并查集 [题解] 1.用tarjan计算强连通分量并缩点,在新图中找入度为0的点的个数就是答案. 但是,会爆内存(题目内存限制64MB). 2.用并查集,最后从1到n统计fa[i ...

  3. 【BZOJ】1535: [POI2005]Sza-Template

    题意 给一个串\(s(1 \le |s| \le 500000)\),求一个最长的串,使得这个串能覆盖整个串(可以重叠). 分析 首先这个串肯定是前缀也肯定是后缀. 题解 对串kmp后,建立\(fai ...

  4. 【BZOJ】3052: [wc2013]糖果公园

    http://www.lydsy.com/JudgeOnline/problem.php?id=3052 题意:n个带颜色的点(m种),q次询问,每次询问x到y的路径上sum{w[次数]*v[颜色]} ...

  5. 【BZOJ】3319: 黑白树

    http://www.lydsy.com/JudgeOnline/problem.php?id=3319 题意:给一棵n节点的树(n<=1e6),m个操作(m<=1e6),每次操作有两种: ...

  6. 【BZOJ】3319: 黑白树(并查集+特殊的技巧/-树链剖分+线段树)

    http://www.lydsy.com/JudgeOnline/problem.php?id=3319 以为是模板题就复习了下hld............................. 然后n ...

  7. 【BZOJ】1013: [JSOI2008]球形空间产生器sphere

    [BZOJ]1013: [JSOI2008]球形空间产生器sphere 题意:给n+1个n维的点的坐标,要你求出一个到这n+1个点距离相等的点的坐标: 思路:高斯消元即第i个点和第i+1个点处理出一个 ...

  8. 【BZOJ】1002:轮状病毒(基尔霍夫矩阵【附公式推导】或打表)

    Description 轮状病毒有很多变种,所有轮状病毒的变种都是从一个轮状基产生的.一个N轮状基由圆环上N个不同的基原子和圆心处一个核原子构成的,2个原子之间的边表示这2个原子之间的信息通道.如下图 ...

  9. 【BZOJ】【3083】遥远的国度

    树链剖分/dfs序 其实过了[BZOJ][4034][HAOI2015]T2以后就好搞了…… 链修改+子树查询+换根 其实静态树的换根直接树链剖分就可以搞了…… 因为其实只有一样变了:子树 如果roo ...

随机推荐

  1. Python无类再理解--metaclass,type

    上次理解过一次,时间久了,就忘了.. 再学习一次.. http://blog.jobbole.com/21351/ ======================= 但是,Python中的类还远不止如此 ...

  2. python列表分组的技巧

    今天项目上需要到的. 如,将并行部署的机器分批次. 一次十台机器,如果分3次并行部署,则第一次123,第二次456,第三次789,第四次10. #coding=utf-8 a=[1,2,3,4,5,6 ...

  3. 攻城狮在路上(叁)Linux(十五)--- 文件与目录的默认权限与隐藏权限

    一.文件默认权限:umask <==需要被减去的权限. 1.umask指的是当前用户在新建文件或者目录时的默认权限,如0022; 2.默认情况下,用户创建文件的最大权限为666; 创建目录的最大 ...

  4. TCP 连接中断的判断

    关于TCP 3次握手,4次分手就不多说了.如何判断TCP连接是否中断是个大问题. 1 TCPKeep-alive 机制 并不是TCP 规范的一部分,实现细节差距大. 2 使用heartbeat 检测 ...

  5. APUE包含头文件"apue.h"问题

    下载源码 从unix高级编程书籍官网下载书籍的上的所有源码. wget http://www.apuebook.com/src.tar.gz 解压这个文件 tar -zxvf src.tar.gz 解 ...

  6. POJ 1163:The Triangle

    Description 7 3 8 8 1 0 2 7 4 4 4 5 2 6 5 (Figure 1) Figure 1 shows a number triangle. Write a progr ...

  7. unity3D与网页的交互---做项目的一点总结

    (来自博客园) 由于项目需要,要求用unity来展示三维场景,并在三维中能够方便的查询数据库等.一开始尝试在unity中直接连接数据库,当时连的xml,然而每次发布成网页后都会出现路径找不到等问题,所 ...

  8. 建造者模式与原型模式/builder模式与prototype模式/创建型模式

    建造者模式 定义 用于简化复杂对象的创建 JDK中的建造者模式 java.lang.StringBuilder中的append()方法,每次调用后返回修改后的对象本身. public StringBu ...

  9. 内容分发网络CDN(互联网技术)

    内容分发网络(互联网技术)CDN的全称是Content Delivery Network,即内容分发网络.其基本思路是尽可能避开互联网上有可能影响数据传输速度和稳定性的瓶颈和环节,使内容传输的更快.更 ...

  10. loj 1030概率dp

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1030 思路:一直以来对这种概率题都挺感冒的=.=......还是说一下思路吧,dp[i ...