[Wc2007]剪刀石头布

题目大意https://www.lydsy.com/JudgeOnline/problem.php?id=2597


题解

发现直接求三元环不好求,我们考虑任选三个点不是三元环的个数。

这样的话,必定是有一个点被其余两个点指,我们就根据这个来求。

又发现,最后的答案之和所有点的度数有关。

就是,$\sum C_{d_i}^{2}$。

紧接着,因为度数和是一定的。而且已经有了一些边。

现在就是有固定的度数可以分配,每个点有一个分配上限,怎么分配最少?

发现一个事,就是$C_{d_i + 1}^{2} - C_{d_i} ^ {2} < C_{d_i + 2}^{2} - C_{d_i + 1}^{2}$。

根据这个性质,我们就可以暴力连边费用流了。

我们就把权值的差当做费用,它肯定会从低往高走因为是最小费用,满足题意。

代码

#include <bits/stdc++.h>

#define N 11010 

#define M 500010 

using namespace std;

int head[N], to[M], pre[N], nxt[M], val[M], cst[M], tot = 1;

int S = N - 1, T = N - 2, dis[N];

bool vis[N];

queue <int> q;

char *p1, *p2, buf[100000];

#define nc() (p1 == p2 && (p2 = (p1 = buf) + fread(buf, 1, 100000, stdin), p1 == p2) ? EOF : *p1 ++ )

int rd() {
int x = 0;
char c = nc();
while (c < 48) {
c = nc();
}
while (c > 47) {
x = (((x << 2) + x) << 1) + (c ^ 48), c = nc();
}
return x;
} inline void add2(int x, int y, int z, int w) {
to[ ++ tot] = y;
nxt[tot] = head[x];
val[tot] = z;
cst[tot] = w;
head[x] = tot;
} inline void add(int x, int y, int z, int w) {
add2(x, y, z, w);
add2(y, x, 0, -w);
} bool spfa() {
while (!q.empty()) {
q.pop();
}
memset(pre, 0, sizeof pre);
memset(dis, 0x3f, sizeof dis);
dis[S] = 0;
q.push(S);
while (!q.empty()) {
int x = q.front();
q.pop();
vis[x] = false;
for (int i = head[x]; i; i = nxt[i]) {
if (dis[to[i]] > dis[x] + cst[i] && val[i]) {
dis[to[i]] = dis[x] + cst[i];
pre[to[i]] = i ^ 1;
if (!vis[to[i]]) {
vis[to[i]] = true;
q.push(to[i]);
}
}
}
}
return pre[T];
} int mincost() {
int re = 0;
while (spfa()) {
int mdl = 0x3f3f3f3f;
for (int i = T; i != S; i = to[pre[i]]) {
mdl = min(mdl, val[pre[i] ^ 1]);
}
for (int i = T; i != S; i = to[pre[i]]) {
val[pre[i] ^ 1] -= mdl;
val[pre[i]] += mdl;
re += mdl * cst[pre[i] ^ 1];
}
}
return re;
} int A[110][110], id[110][110]; int main() {
int n = rd();
for (int i = 1; i <= n; i ++ ) {
for (int j = 1; j <= n; j ++ ) {
A[i][j] = rd();
}
}
int cnt = n;
for (int i = 1; i <= n; i ++ ) {
for (int j = i + 1; j <= n; j ++ ) {
if (A[i][j] == 2) {
cnt ++ ;
add(cnt, i, 1, 0);
id[i][j] = tot - 1;
add(cnt, j, 1, 0);
id[j][i] = tot - 1;
add(S, cnt, 1, 0);
}
}
}
// cout << tot << ' ' << cnt << endl ;
for (int i = 1; i <= n; i ++ ) {
int c = 0;
for (int j = 1; j <= n; j ++ ) {
if (A[i][j] == 1) {
c ++ ;
}
}
if (c) {
add(S, i, c, 0);
}
}
for (int i = 1; i <= n; i ++ ) {
for (int j = 1; j <= n; j ++ ) {
add(i, T, 1, j * (j - 1) / 2 - (j - 1) * (j - 2) / 2);
}
}
int ans = mincost();
ans = n * (n - 1) * (n - 2) / 6 - ans;
cout << ans << endl ;
for (int i = 1; i <= n; i ++ ) {
for (int j = 1; j <= n; j ++ ) {
if (A[i][j] != 2) {
printf("%d ", A[i][j]);
}
else {
if (i == j) {
printf("0 ");
}
else {
if (!val[id[i][j]]) {
printf("1 ");
}
else {
printf("0 ");
}
}
}
}
puts("");
}
return 0;
}
/*
6
0 2 1 0 2 1
2 0 0 2 2 2
0 1 0 2 1 1
1 2 2 0 1 2
2 2 0 0 0 2
0 2 0 2 2 0 */

小结:这个费用流很巧妙。就是有先后的选取过程,但是保证后面的代价比前面的代价大,我们可以暴力建边。

[bzoj2597][Wc2007]剪刀石头布_费用流的更多相关文章

  1. BZOJ2597 [Wc2007]剪刀石头布 【费用流】

    题目链接 BZOJ2597 题解 orz思维差 既然是一张竞赛图,我们选出任意三个点都可能成环 总方案数为 \[{n \choose 3}\] 如果三个点不成环,会发现它们的度数是确定的,入度分别为\ ...

  2. bzoj2597: [Wc2007]剪刀石头布(费用流)

    传送门 不得不说这思路真是太妙了 考虑能构成三元组很难,那我们考虑不能构成三元组的情况是怎么样 就是说一个三元组$(a,b,c)$,其中$a$赢两场,$b$赢一场,$c$没有赢 所以如果第$i$个人赢 ...

  3. BZOJ2597 WC2007剪刀石头布(费用流)

    考虑使非剪刀石头布情况尽量少.设第i个人赢了xi场,那么以i作为赢家的非剪刀石头布情况就为xi(xi-1)/2种.那么使Σxi(xi-1)/2尽量小即可. 考虑网络流.将比赛建成一排点,人建成一排点, ...

  4. BZOJ_2661_[BeiJing wc2012]连连看_费用流

    BZOJ_2661_[BeiJing wc2012]连连看_费用流 Description 凡是考智商的题里面总会有这么一种消除游戏.不过现在面对的这关连连看可不是QQ游戏里那种考眼力的游戏.我们的规 ...

  5. BZOJ_1834_[ZJOI2010]network 网络扩容_费用流

    BZOJ_1834_[ZJOI2010]network 网络扩容_费用流 题意: 给定一张有向图,每条边都有一个容量C和一个扩容费用W.这里扩容费用是指将容量扩大1所需的费用. 求:  1.在不扩容的 ...

  6. BZOJ_1877_[SDOI2009]晨跑_费用流

    BZOJ_1877_[SDOI2009]晨跑_费用流 题意: Elaxia最近迷恋上了空手道,他为自己设定了一套健身计划,比如俯卧撑.仰卧起坐等 等,不过到目前为止,他 坚持下来的只有晨跑. 现在给出 ...

  7. BZOJ_1070_[SCOI2007]修车_费用流

    BZOJ_1070_[SCOI2007]修车_费用流 Description 同一时刻有N位车主带着他们的爱车来到了汽车维修中心.维修中心共有M位技术人员,不同的技术人员对不同 的车进行维修所用的时间 ...

  8. 【BZOJ-2597】剪刀石头布 最小费用最大流

    2597: [Wc2007]剪刀石头布 Time Limit: 20 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 1016  Solved:  ...

  9. [WC2007]剪刀石头布(最大流)

    洛古 一句话题意:给定一张图,每两点之间有一条有向边或无向边,把所有无向边定向,使图中三元环个数尽量多 因为原图是一个完全图,假设图中任意三点都能构成三元环,那么途中三元环的个数为:\(\binom{ ...

随机推荐

  1. ActiveMQ介绍

    一.背景 中间件 由于业务的不同.技术的发展.硬件和软件的选择有所差别,导致了异构组件或应用并存的局面.要使这些异构的组件协同工作,一个有效的方式就是提供一个允许它们进行通信的层,该层即为中间件. 在 ...

  2. 百度智能api接口汇总

    一:自然语言处理 # -*- coding: utf-8 -*- # @Author : FELIX # @Date : 2018/5/18 9:47 # pip install baidu-aip ...

  3. Eclise快捷键(最全)

    Eclipse常用快捷键 Eclipse常用快捷键 1几个最重要的快捷键 代码助手:Ctrl+Space(简体中文操作系统是Alt+/) 快速修正:Ctrl+1 单词补全:Alt+/ 打开外部Java ...

  4. Appium基础教程

    目录 Appium教程 Appium简介 App自动化测试工具对比 Appium实现原理 环境搭建 Andorid介绍 基本架构 常见布局/视图 基本控件 控件常见属性 Adb介绍 Adb常用命令 A ...

  5. PHP 根据两个坐标计算距离 圆形围栏的计算

    可以应用于圆形电子围栏入 出围栏计算 圆形电子围栏的计算方式是: 根据圆心坐标和当前检查的坐标进行距离计算,如果距离长度超出围栏的半径,则判定为出围栏,反之是在围栏之内 <?php /** * ...

  6. 2017 ZSTU寒假排位赛 #4

    题目链接:https://vjudge.net/contest/148543#overview. A题:n个罪犯,每个人有一个犯罪值,现在要从里面选出连续的c个人,每个人的犯罪值都不能超过t,问选法的 ...

  7. Java中可变参数

    从java5开始出现了可变参数,这是对java方法及数组的拓展! 方法中可以接受的参数不再是固定个数的,而是随着具体需求传递的多少来决定. 定义格式: 返回值类型  方法名(参数类型 ... 形式参数 ...

  8. flask 部署

    学习 Flask,写完一个 Flask 应用需要部署的时候,就想着折腾自己的服务器.根据搜索的教程照做,对于原理一知半解,磕磕碰碰,只要运行起来了,谢天谢地然后不再折腾了,到下一次还需要部署时,这样的 ...

  9. 【平台中间件】为什么用etcd而不用ZooKeeper?从应用场景到实现原理的全方位解读

    前言 博主在工作过程中经常接触到ETCD,搜索相关资料的时候发现排名最高的是一篇图片全是404的转载文章,后来看到了原文,感觉有义务让更多的人看到这样的精品文章,所以进行了转载. 原文发布在infoQ ...

  10. openwrt系统源码地址

    https://dev.openwrt.org/wiki/GetSource http://www.openwrtdl.com/wordpress/openwrt-full-tutorial