【BZOJ3232】圈地游戏(分数规划,网络流)

题面

BZOJ

题解

很神仙的一道题。

首先看到最大化的比值很容易想到分数规划。现在考虑分数规划之后怎么计算贡献。

首先每条边的贡献就变成了\(mid*C\),这个显然啊。考虑一个封闭图形如何计算答案。

发现被计算入答案的边一定是一侧有一个格子被圈进去了,另外一侧的格子没有被圈进去。那么这很像一个最小割。假设格子和源点相连表示被选进了答案,和汇点相连表示在答案以外。那么很明显把一条边两侧的格子给连起来,流量为\(mid*C\)。怎么越来越像一个最大权闭合子图了。。。

那么每个点如果要和源点断开就必定会减小它自身的权值,所以\(S\)向每个格子连格子价值的边。然而忽然发现没有东西和汇点连在一起?因为没有必须不选的格子?不,显然是有的,我们有的边显然和界外相连,所以界外必须额外新建一个点,那么从界外向汇点连一条容量为\(inf\)的边表示强制不能断开,即必须不选。这样一来,拿总的点权减去最小割判定二分即可。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define inf 1000000000
#define MAX 55
#define MAXL MAX*MAX*4
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
struct Line{int v,next;double w;}e[MAXL<<1];
int h[MAX*MAX],cnt=2;
inline void Add(int u,int v,double w)
{
e[cnt]=(Line){v,h[u],w};h[u]=cnt++;
e[cnt]=(Line){u,h[v],w};h[v]=cnt++;
}
int n,m,tot,Sum;
int S,T,Out;
int bh[MAX][MAX],V[MAX][MAX],E1[MAX][MAX],E2[MAX][MAX];
void BuildEdge(double mid)
{
memset(h,0,sizeof(h));cnt=2;
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)
Add(S,bh[i][j],V[i][j]);
for(int j=1;j<=m;++j)Add(bh[1][j],Out,mid*E1[1][j]);
for(int i=2;i<=n;++i)
for(int j=1;j<=m;++j)
Add(bh[i-1][j],bh[i][j],mid*E1[i][j]);
for(int j=1;j<=m;++j)Add(bh[n][j],Out,mid*E1[n+1][j]);
for(int i=1;i<=n;++i)Add(bh[i][1],Out,mid*E2[i][1]);
for(int i=1;i<=n;++i)
for(int j=2;j<=m;++j)
Add(bh[i][j-1],bh[i][j],mid*E2[i][j]);
for(int i=1;i<=n;++i)Add(bh[i][m],Out,mid*E2[i][m+1]);
}
int level[MAX*MAX],cur[MAX*MAX];
bool bfs()
{
memset(level,0,sizeof(level));
queue<int> Q;level[S]=1;Q.push(S);
while(!Q.empty())
{
int u=Q.front();Q.pop();
for(int i=h[u];i;i=e[i].next)
if(fabs(e[i].w)>1e-5&&!level[e[i].v])
level[e[i].v]=level[u]+1,Q.push(e[i].v);
}
return level[T];
}
double dfs(int u,double flow)
{
if(u==T||flow<1e-5)return flow;
double ret=0;
for(int &i=cur[u];i;i=e[i].next)
{
int v=e[i].v;double d;
if(fabs(e[i].w)>1e-5&&level[v]==level[u]+1)
{
d=dfs(v,min(flow,e[i].w));
ret+=d;flow-=d;
e[i].w-=d;e[i^1].w+=d;
}
}
if(ret<1e-5)level[u]=0;
return ret;
}
double Dinic()
{
double ret=0;
while(bfs())
{
for(int i=S;i<=T;++i)cur[i]=h[i];
ret+=dfs(S,inf);
}
return ret;
}
int main()
{
n=read();m=read();
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)bh[i][j]=++tot,V[i][j]=read();
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)Sum+=V[i][j];
S=0;T=Out=tot+1;
for(int i=1;i<=n+1;++i)
for(int j=1;j<=m;++j)E1[i][j]=read();
for(int i=1;i<=n;++i)
for(int j=1;j<=m+1;++j)E2[i][j]=read();
double l=0,r=50000;
while(r-l>1e-5)
{
double mid=(l+r)/2;
BuildEdge(mid);
if(Sum-Dinic()>1e-5)l=mid;
else r=mid;
}
printf("%.3lf\n",l);
return 0;
}

【BZOJ3232】圈地游戏(分数规划,网络流)的更多相关文章

  1. 【BZOJ3232】圈地游戏 分数规划+最小割

    [BZOJ3232]圈地游戏 Description DZY家的后院有一块地,由N行M列的方格组成,格子内种的菜有一定的价值,并且每一条单位长度的格线有一定的费用. DZY喜欢在地里散步.他总是从任意 ...

  2. BZOJ 3232: 圈地游戏 分数规划+判负环

    3232: 圈地游戏 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 966  Solved: 466[Submit][Status][Discuss] ...

  3. bzoj3232圈地游戏——0/1分数规划+差分建模+判环

    Description DZY家的后院有一块地,由N行M列的方格组成,格子内种的菜有一定的价值,并且每一条单位长度的格线有一定的费用. DZY喜欢在地里散步.他总是从任意一个格点出发,沿着格线行走直到 ...

  4. $BZOJ3232$ 圈地游戏 网络流

    正解:最小割+01分数规划 解题报告: 传送门$QwQ$ 感$jio$这个好像是$NOIp2018$集训的时候$cjk$学长讲01分数规划的时候港的,,,?$QwQ$虽然我还是不会嘤 首先看到这个分数 ...

  5. BZOJ3232: 圈地游戏

    题解: 神题一道... 题解戳这里:http://hi.baidu.com/strongoier/item/0425f0e5814e010265db0095 分数规划可以看这里:http://blog ...

  6. 【题解】 [HNOI2009] 最小圈 (01分数规划,二分答案,负环)

    题目背景 如果你能提供题面或者题意简述,请直接在讨论区发帖,感谢你的贡献. 题目描述 对于一张有向图,要你求图中最小圈的平均值最小是多少,即若一个圈经过k个节点,那么一个圈的平均值为圈上k条边权的和除 ...

  7. BZOJ1486 HNOI2009 最小圈 【01分数规划】

    BZOJ1486 HNOI2009 最小圈 Description 应该算是01分数规划的裸板题了吧..但是第一次写还是遇到了一些困难,vis数组不清零之类的 假设一个答案成立,那么一定可以找到一个环 ...

  8. BZOJ2285 [SDOI2011]保密 【01分数规划 + 网络流】

    题目 现在,保密成为一个很重要也很困难的问题.如果没有做好,后果是严重的.比如,有个人没有自己去修电脑,又没有拆硬盘,后来的事大家都知道了. 当然,对保密最需求的当然是军方,其次才是像那个人.为了应付 ...

  9. 【XSY2718】gift 分数规划 网络流

    题目描述 有\(n\)个物品,买第\(i\)个物品要花费\(a_i\)元.还有\(m\)对关系:同时买\(p_i,q_i\)两个物品会获得\(b_i\)点收益. 设收益为\(B\),花费为\(A\), ...

  10. 2018.09.24 bzoj1486: [HNOI2009]最小圈(01分数规划+spfa判负环)

    传送门 答案只保留了6位小数WA了两次233. 这就是一个简单的01分数规划. 直接二分答案,根据图中有没有负环存在进行调整. 注意二分边界. 另外dfs版spfa判负环真心快很多. 代码: #inc ...

随机推荐

  1. go语言之行--基础部分

    一.数据类型 布尔型 布尔类型 - 由两个预定义常量组成:true.false,默认值为false package main import "fmt" func main() { ...

  2. 20155318 Exp1 PC平台逆向破解(5)M

    20155318 Exp1 PC平台逆向破解(5)M 实践目标 本次实践的对象是一个名为pwn1的linux可执行文件. 该程序正常执行流程是:main调用foo函数,foo函数会简单回显任何用户输入 ...

  3. C++之enum枚举量声明、定义、使用与枚举类详解

    C++之enum枚举量声明.定义.使用与枚举类详解 学习一个东西,首先应该指导它能做什么,其次去知道它怎么去做,最后知道为什么去这么做. 知其然知其所以然.不能冒进 ,一步一步的慢慢来.

  4. Scala学习(二)--- 控制结构和函数

    控制结构和函数 摘要: 本篇主要学习在Scala中使用条件表达式.循环和函数,你会看到Scala和其他编程语言之间一个根本性的差异.在Java或C++中,我们把表达式(比如3+4)和语句(比如if语句 ...

  5. 4字节emoji表情对应的Unicode编码获取和编码转换

    GitHub Flavored Markdown 今天研究了一天Markdown移动端和pc端统一实现方式,由于以前有搞过移动端富文本编辑器,搞Markdown简单多了: 其中GFM的表情语法不错,比 ...

  6. win10 添加项目右键用vscode打开

    1.新建reg文件:在vscode安装目录下新建一个文本文件,然后将文件后缀改为:*.reg,文件名任意,例如:vsCodeOpenFolder.reg. 2.编写文本文件内容.将下面的内容Copy到 ...

  7. 开发认为不是bug,你该如何处理?

    这是软件测试员面试时经常被问到的问题.看了很多答案,个人觉得作为有工作经验的测试人员回答时不能完全照搬标准答案,技术面试官想听的当然不止如此.毕竟这种情况在实际工作中也常常出现,具体问题要具体分析,你 ...

  8. MIT-6.824 MapReduce

    概述 MapReduce是由JeffreyDean提出的一种处理大数据的编程模型,用户定义map和reduce函数,map函数处理原始数据生成一系列键值对中间数据,reduce函数并合相同key的键值 ...

  9. 散列(Hash)表入门

    一.概述 以 Key-Value 的形式进行数据存取的映射(map)结构 简单理解:用最基本的向量(数组)作为底层物理存储结构,通过适当的散列函数在词条的关键码与向量单元的秩(下标)之间建立映射关系 ...

  10. Java serialVersionUID作用和生成

    序列化和反序列化Java是面向对象的语言,与其他语言进行交互(比如与前端js进行http通信),需要把对象转化成一种通用的格式比如json(前端显然不认识Java对象),从对象到json字符串的转换, ...