bzoj 2597 剪刀石头布 —— 拆边费用流
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2597
不合法的三个人之间的关系就是一个人赢了两次;
记 \( deg[i] \) 表示第 \( i \) 个人赢的次数,那么答案就是 \( C_{n}^{3} - \sum\limits_{i=1}^{n} C_{deg[i]}^{2} \)
对于不确定的关系,如果确定 \( x \) 赢,那么答案中减去的不合法情况又会多 \( deg[x] \) 种;
也就是第一次多 \( deg[x] \),第二次多 \( deg[x] + 1 \),第三次多 \( deg[x] + 2 \) ...
如果把这些作为费用,因为每次走费用最小的边,正好是第一次 \( deg[x] \),第二次 \( deg[x] + 1 \)...
于是把每条无向边作为一个点,提供1的流量给两个端点中的一个表示哪个赢,赢的端点 \( x \) 向汇点连一系列费用从 \( deg[x] \) 开始递增的边即可;
WA了一次还以为是写法太麻烦,正准备大改,突然发现输出格式不对...数字中间要加空格的。
代码如下:
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
using namespace std;
int const xn=,xm=1e5+,xxn=,inf=0x3f3f3f3f;
int n,cnt,id[xxn][xxn],dis[xn],hd[xn],to[xm],nxt[xm],ct=,c[xm],w[xm];
int S,T,pre[xn],inc[xn],deg[xxn],num[xxn],sid[xxn][xxn];
bool vis[xn];
queue<int>q;
struct N{int u,v;}ed[xm];
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
void add(int x,int y,int z,int f){to[++ct]=y; nxt[ct]=hd[x]; hd[x]=ct; w[ct]=z; c[ct]=f;}
bool bfs()
{
memset(dis,0x3f,sizeof dis);
memset(pre,,sizeof pre);
memset(inc,0x3f,sizeof inc);
dis[S]=; q.push(S); vis[S]=;
while(q.size())
{
int x=q.front(); q.pop(); vis[x]=;
for(int i=hd[x],u;i;i=nxt[i])
if(dis[u=to[i]]>dis[x]+w[i]&&c[i])
{
dis[u]=dis[x]+w[i];
pre[u]=i; inc[u]=min(inc[x],c[i]);
if(!vis[u])vis[u]=,q.push(u);
}
}
return dis[T]!=inf;
}
void up()
{
int x=T;
while(x!=S)
{
int i=pre[x];
c[i]-=inc[T]; c[i^]+=inc[T];
x=to[i^];
}
}
int main()
{
n=rd();
for(int i=;i<=n;i++)
for(int j=,d;j<=n;j++)
{
d=rd(); sid[i][j]=d; if(i>=j)continue;
if(d==)deg[i]++; else if(d==) deg[j]++;
else
{
id[i][j]=id[j][i]=++cnt;
ed[cnt].u=i,ed[cnt].v=j;
num[i]++; num[j]++;//possible win
}
}
S=; T=cnt+n+;
for(int i=;i<=cnt;i++)
{
add(S,i,,),add(i,S,,);
add(i,ed[i].u+cnt,,),add(ed[i].u+cnt,i,,);//+cnt
add(i,ed[i].v+cnt,,),add(ed[i].v+cnt,i,,);
}
int ans=;
for(int i=;i<=n;i++)ans+=deg[i]*(deg[i]-)/;
for(int i=cnt+;i<=cnt+n;i++)
for(int j=;j<=num[i-cnt];j++)//-cnt
add(i,T,deg[i-cnt]+j-,),add(T,i,,);//-cnt
while(bfs())ans+=dis[T]*inc[T],up();//*
printf("%d\n",n*(n-)*(n-)/-ans);
for(int i=;i<=n;i++,puts(""))
for(int j=;j<=n;j++)
{
if(sid[i][j]!=){printf("%d ",sid[i][j]); continue;}
int x=id[i][j],t;
for(int k=hd[x],u;k;k=nxt[k])
if(to[k]==i+cnt&&!c[k]){t=i; break;}//+cnt
else if(to[k]==j+cnt&&!c[k]){t=j; break;}
if(t==i)printf("1 "),sid[j][i]=;
else printf("0 "),sid[j][i]=;
}
return ;
}
bzoj 2597 剪刀石头布 —— 拆边费用流的更多相关文章
- [bzoj 1449] 球队收益(费用流)
		[bzoj 1449] 球队收益(费用流) Description Input Output 一个整数表示联盟里所有球队收益之和的最小值. Sample Input 3 3 1 0 2 1 1 1 1 ... 
- [bzoj2597][Wc2007]剪刀石头布_费用流
		[Wc2007]剪刀石头布 题目大意:https://www.lydsy.com/JudgeOnline/problem.php?id=2597 题解: 发现直接求三元环不好求,我们考虑任选三个点不是 ... 
- 【BZOJ4930】棋盘 拆边费用流
		[BZOJ4930]棋盘 Description 给定一个n×n的棋盘,棋盘上每个位置要么为空要么为障碍.定义棋盘上两个位置(x,y),(u,v)能互相攻击当前仅 当满足以下两个条件: 1:x=u或y ... 
- bzoj 1070: [SCOI2007]修车 费用流
		1070: [SCOI2007]修车 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 2785 Solved: 1110[Submit][Status] ... 
- BZOJ 3171 循环格(费用流)
		题意 一个循环格就是一个矩阵,其中所有元素为箭头,指向相邻四个格子.每个元素有一个坐标(行,列),其中左上角元素坐标为(0,0).给定一个起始位置(r,c),你可以沿着箭头防线在格子间行走.即如果(r ... 
- BZOJ 1070 修车  【费用流】
		Description 同一时刻有N位车主带着他们的爱车来到了汽车维修中心.维修中心共有M位技术人员,不同的技术人员对不同 的车进行维修所用的时间是不同的.现在需要安排这M位技术人员所维修的车及顺序, ... 
- BZOJ 1930 吃豆豆(费用流)
		首先这题的两条线不相交的限制可以去掉,因为如果相交的话把点换一换是不影响最终结果的. 剩下的费用流建图是显然的,把点拆为两个,建立超级源点s和源点ss汇点t,连边(s,ss,2,0). 对于每个点,连 ... 
- BZOJ 1927 星际竞速(费用流)
		考虑费用流,题目要求走n个点都走完且恰好一次,显然流量的限制为n. 建立源点s和汇点t,并把每个星球拆成两个点i和i',分别表示已到达该点和经过该点. 对于能力爆发,建边(s,i',1,w). 对应高 ... 
- BZOJ 1221 软件开发(费用流)
		容易看出这是显然的费用流模型. 把每天需要的餐巾数作为限制.需要将天数拆点,x’表示每天需要的餐巾,x’’表示每天用完的餐巾.所以加边 (s,x',INF,0),(x'',t,INF,0). 餐巾可以 ... 
随机推荐
- vim对光标所在的数字进行增减
			真是vim会在不经意间给你惊喜...... 现在发现把光标移到某数字的上方,c-a是加1, c-x是减1 当时真有点众里寻他千百度的感觉 
- zookeeper(二): Curator vs  zkClient
			目录 zookeeper Curator zkClient 客户端对比 写在前面 1.1. zookeeper应用开发 1.1.1. ZkClient简介 1.1.2. Curator简介 写在最后 ... 
- Linux c编程:同步属性
			就像线程具有属性一样,线程的同步对象(如互斥量.读写锁.条件变量.自旋锁和屏障)也有属性 1.互斥量属性 用pthread_mutexattr_init初始化pthread_mutexattr_t结构 ... 
- ubuntu安装pip和python
			安装pip2sudo apt-get install pip 这样安装的是pip2不支持Python3.x,可以使用如下命令安装pip3 sudo apt-get install python3-pi ... 
- GIT笔记:GITHUB教程【官方自译版】
			GIT笔记:将项目发布到GITHUB GITHUB是什么 GitHub是版本控制和协作的代码托管平台.它可以让你和其他人在任何地方一起工作. 1.创建一个新的仓库 存储库通常用于组织单个项目.存储库可 ... 
- eclipse---个人设置
			window---- preferences -----修改背景颜色 -----修改字体 ----修改窗口主题 ----设置编码 -----设置编译环境 ----设置web项目JDK编译的版本 --- ... 
- chorme 插件
			json-handle: json可视化工具 开发中需要用到json,在浏览器显示的json非常乱,难以理解.有没有让人一目了然的工具,让json看起来非常直观呢,json-handle随之而出,包含 ... 
- python cookbook 字符串和文本
			使用多个界定符分隔字符串 import re line = 'asdf fjdk; afed, fjek,asdf, foo' print(re.split(r'[;,\s]\s*', line)) ... 
- Data Structure Binary Tree: Boundary Traversal of binary tree
			http://www.geeksforgeeks.org/boundary-traversal-of-binary-tree/ #include <iostream> #include & ... 
- Spark Structured Streaming框架(4)之窗口管理详解
			1. 结构 1.1 概述 Structured Streaming组件滑动窗口功能由三个参数决定其功能:窗口时间.滑动步长和触发时间. 窗口时间:是指确定数据操作的长度: 滑动步长:是指窗口每次向前移 ... 
