【BZOJ-2597】剪刀石头布 最小费用最大流
2597: [Wc2007]剪刀石头布
Time Limit: 20 Sec Memory Limit: 128 MBSec Special Judge
Submit: 1016 Solved: 477
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
0 1 2
0 0 2
2 2 0
Sample Output
0 1 0
0 0 1
1 0 0
HINT
100%的数据中,N≤ 100。
Source
Solution
这道题,直接最大化剪刀石头布的数量很麻烦,所以可以考虑最小化不是剪刀石头布的数量,所以可以考虑利用最小费用流。
考虑如果一个人赢了$x_{i}$场,那么最后不是剪刀石头布的数量就是$\sum \frac{x_{i}*(x_{i}-1)}{2}$,所以就是最小化这个量.
建图很显然,对于一场比赛$<i,j>$新建一个节点,如果$<i,j>$确定,则向胜利一方连容量$1$,费用$0$;如果不确定则都连容量$1$,费用$0$;
然后源点$S$向每场比赛连容量$1$,费用$0$;每个人向$T$连边容量为$N$,限制费用.
问题在于如何限制费用另其满足对答案的贡献为$\frac{x_{i}*(x_{i}-1)}{2}$,然后我想到对于求$\sum_{i=1}^{N}=\frac{N*(N+1)}{2}$,然后如果令$N=N-1$,答案正好是$\frac{N*(N-1)}{2}$,所以就可以拆成$N$条边,费用依次是$0..N-1$
然后就可以了.最后方案统计一下就好,即统计比赛$<i,j>$流向$i$还是流向$j$.
Code
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<queue>
using namespace std;
inline int read()
{
int x=0,f=1; char ch=getchar();
while (ch<'0' || ch>'9') {if (ch=='-') f=-1; ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0'; ch=getchar();}
return x*f;
}
#define MAXN 11000
#define MAXM 500010
#define INF 0x7fffffff
int N,mp[110][110],id[110][110],e[10100][110];
struct EdgeNode{int next,to,from,cap,cost;}edge[MAXM<<1];
int head[MAXN],cnt=1;
inline void AddEdge(int u,int v,int w,int c) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v; edge[cnt].from=u; edge[cnt].cost=c; edge[cnt].cap=w;}
inline void InsertEdge(int u,int v,int w,int c) {AddEdge(u,v,w,c); AddEdge(v,u,0,-c);}
int mark[MAXM],dis[MAXM],Flow,Cost,S,T;
inline bool SPFA()
{
for (int i=S; i<=T; i++) dis[i]=INF,mark[i]=0;
queue<int>q;
q.push(S); dis[S]=0; mark[S]=1;
while (!q.empty())
{
int now=q.front(); q.pop(); mark[now]=0;
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].cap && dis[edge[i].to]>dis[now]+edge[i].cost)
{
dis[edge[i].to]=dis[now]+edge[i].cost;
if (!mark[edge[i].to]) q.push(edge[i].to),mark[edge[i].to]=1;
}
}
return dis[T]!=INF;
}
inline int DFS(int loc,int low)
{
mark[loc]=1;
if (loc==T) return low;
int used=0,w;
for (int i=head[loc]; i; i=edge[i].next)
if (edge[i].cap && !mark[edge[i].to] && dis[edge[i].to]==dis[loc]+edge[i].cost)
{
w=DFS(edge[i].to,min(edge[i].cap,low-used));
edge[i].cap-=w; edge[i^1].cap+=w; used+=w; Cost+=w*edge[i].cost;
if (low==used) return used;
}
return used;
}
inline int ZKW()
{
int re=0;
while (SPFA())
{
mark[T]=1;
while (mark[T])
{
for (int i=S; i<=T; i++) mark[i]=0;
re+=DFS(S,INF);
}
}
return re;
}
int main()
{
N=read();
for (int i=1; i<=N; i++)
for (int j=1; j<=N; j++) mp[i][j]=read();
int ID=N;
for (int i=1; i<=N; i++)
for (int j=1; j<=N; j++) id[i][j]=++ID;
S=0; T=ID+1;
for (int i=1; i<=N; i++)
for (int j=i+1; j<=N; j++)
if (i!=j)
{
InsertEdge(S,id[i][j],1,0);
switch (mp[i][j])
{
case 1: InsertEdge(id[i][j],i,1,0); break;
case 0: InsertEdge(id[i][j],j,1,0); break;
case 2: InsertEdge(id[i][j],i,1,0); e[id[i][j]][i]=cnt-1;
InsertEdge(id[i][j],j,1,0); e[id[i][j]][j]=cnt-1; break;
}
}
for (int i=1; i<=N; i++)
for (int j=1; j<=N; j++) InsertEdge(i,T,1,j-1);
ZKW();
printf("%d\n",N*(N-2)*(N-1)/6-Cost);
for (int i=1; i<=N; i++)
for (int j=i+1; j<=N; j++)
if (mp[i][j]==2)
{
mp[i][j]=edge[e[id[i][j]][i]].cap==0;
mp[j][i]=mp[i][j]^1;
}
for (int i=1; i<=N; i++,puts(""))
for (int j=1; j<=N; j++) printf("%d ",mp[i][j]);
return 0;
}
【BZOJ-2597】剪刀石头布 最小费用最大流的更多相关文章
- bzoj 2597 剪刀石头布 —— 拆边费用流
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2597 不合法的三个人之间的关系就是一个人赢了两次: 记 \( deg[i] \) 表示第 \ ...
- BZOJ2597 [Wc2007]剪刀石头布(最小费用最大流)
题目大概是说n个人两两进行比赛,问如何安排几场比赛的输赢使得A胜B,B胜C,C胜A这种剪刀石头布的三元组最多. 这题好神. 首先,三元组总共有$C_n^3$个 然后考虑最小化不满足剪刀石头布条件的三元 ...
- BZOJ 2668 [cqoi2012]交换棋子 | 最小费用最大流
传送门 BZOJ 2668 题解 同时分别限制流入和流出次数,所以把一个点拆成三个:入点in(x).中间点mi(x).出点ou(x). 如果一个格子x在初始状态是黑点,则连(S, mi(x), 1, ...
- BZOJ 3876 [AHOI/JSOI2014]支线剧情 (最小费用可行流)
题面:洛谷传送门 BZOJ传送门 题目大意:给你一张有向无环图,边有边权,让我们用任意条从1号点开始的路径覆盖这张图,需要保证覆盖完成后图内所有边都被覆盖至少一次,求覆盖路径总长度的最小值 最小费用可 ...
- 【BZOJ】1221: [HNOI2001] 软件开发(最小费用最大流)
http://www.lydsy.com/JudgeOnline/problem.php?id=1221 先吐槽一下,数组依旧开小了RE:在spfa中用了memset和<queue>的版本 ...
- BZOJ 1927 星际竞速(最小费用最大流)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1927 题意:一个图,n个点.对于给出的每条边 u,v,w,表示u和v中编号小的那个到编号 ...
- BZOJ 1061 志愿者招募(最小费用最大流)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1061 题意:申奥成功后,布布经过不懈努力,终于 成为奥组委下属公司人力资源部门的主管.布 ...
- bzoj 1877 [SDOI2009]晨跑(最小费用最大流)
Description Elaxia最近迷恋上了空手道,他为自己设定了一套健身计划,比如俯卧撑.仰卧起坐等 等,不过到目前为止,他坚持下来的只有晨跑. 现在给出一张学校附近的地图,这张地图中包含N个十 ...
- bzoj 1927 [Sdoi2010]星际竞速(最小费用最大流)
1927: [Sdoi2010]星际竞速 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 1576 Solved: 954[Submit][Statu ...
随机推荐
- 【移动前端开发实践】从无到有(统计、请求、MVC、模块化)H5开发须知
前言 不知不觉来百度已有半年之久,这半年是996的半年,是孤军奋战的半年,是跌跌撞撞的半年,一个字:真的是累死人啦! 我所进入的团队相当于公司内部创业团队,人员基本全部是新招的,最初开发时连数据库都没 ...
- Jquery取得iframe中元素的几种方法
[jquery]获取iframe中的body元素: $("iframe").contents().find("body").html(); [使用jquery操 ...
- iOS多线程之7.NSOperation的初识
NSOperation和GCD一样,不用我们管理线程的生命周期,加锁等问题,只要把操作封装进NSOperation中,系统会自动帮我们创建线程,执行操作.而且他是面向对象的,我们看起来更容易理解,使用 ...
- Collection和Collections的区别?
Collection 是接口(Interface),是集合类的上层接口. Collections是类(Class),集合操作的工具类,服务于Collection框架.它是一个算法类,提供一系列静态方法 ...
- docker通过iptables修改或新增镜像映射端口
443 8088 22 端口是初始映射端口 [root@SERVER ~]# docker ps CONTAINER ID IMAGE COMMAND CREATED STATUS PORTS NAM ...
- 简历生成平台项目开发-STEP5初步界面demo实现
谭卓因为暑期实习,去杭州实习了,走之前在git上上传了一些文档(https://github.com/USTC-CV-creator/),项目到目前为止,前端demo已经做好,后台接收请求生成PDF部 ...
- DBCC CHECKDB 遭遇Operating system error 112(failed to retrieve text for this error. Reason: 15105) encountered
我们一个SQL Server服务器在执行YourSQLDBa的作业YourSQLDba_FullBackups_And_Maintenance时遇到了错误: Exec YourSQLDba.Maint ...
- Oracle学习笔记十三 触发器
简介 触发器是当特定事件出现时自动执行的存储过程,特定事件可以是执行更新的DML语句和DDL语句,触发器不能被显式调用. 触发器的功能: 1.自动生成数据 2.自定义复杂的安全权限 3.提供审计和 ...
- tomcat 应用部署方式(转)
tomcat部署web应用的三种方式 1.直接放到Webapps目录下 Tomcat的Webapps目录是Tomcat默认的应用目录,当服务器启动时,会加载所有这个目录下的应用.也可以将JSP程 ...
- Ubuntu apt 常用命令
APT(the Advanced Packaging Tool)是Ubuntu 软件包管理系统的高级界面,Ubuntu 是基于Debian的,APT由几个名字以“apt-”打头的程序组成.apt-g ...