【BZOJ 3232】圈地游戏 二分+SPFA判环/最小割经典模型
最小割经典模型指的是“一堆元素进行选取,对于某个元素的取舍有代价或价值,对于某些对元素,选取后会有额外代价或价值”的经典最小割模型,建立倒三角进行最小割。
这个二分是显然的,一开始我也是想到了最小割的那个模型的但是我觉得他会不是一个圈我就否掉了,但是仔细想想的话会发现,如果是这样的话所得到的答案一定小于等于一个圈的答案(浓度),所以我们可定会得到最终答案,所以这样做是可以的,所以说要有宽松得正解的意识(泥沙俱下但沙子不影响我泥)。当时我否掉最小割以后就立马去想费用流了,然后想到建图后发现那样建图虽然不好跑费用流,但是SPFA判环还是很劲的,所以我就判了一发环。
在这里就顺便说一下SPFA判负(正)环吧。DFS的话就是判断一个点是否重复出现在DFS路径中,他有一个优化(没看呢),就叫他DFS+吧。然后他还有BFS版的,就是判断一个点是否重复入队n次(点数),但是不能判断是否被更新n次,这样有可能会出错(不用重边就可以做到)(也许可以分析是否可行但是不会很简单而且很难考虑周全),并且这两种方法的时间复杂度有些时候差距并不大只不过是一个常数。网上还有人说是进队次数大于入度,这个经试验证明是扯淡。还有另一种做法是判断到达次点的最短路径的边数等于n这个不仅很对还很快,就叫他BFS+吧。
对于这道题DFS会T,然而DFS+,BFS,以及BFS+均可过,而BFS+表现最优。这说明虽然找最短路方面BFS_SPFA找最短路比DFS_SPFA要好,但是在判环方面并不是DFS一定优于BFS,比如这道题,所以说BFS大法吼。
#include <cstdio>
#include <cstring>
#include <algorithm>
#define pos(a,b) (((a)-1)*(m+1)+(b))
typedef long double db;
const int N=;
const int P=N*N;
const int E=P*;
const db oo=-1e18;
const db eps=1e-;
const db ans_eps=1e-;
struct V{
int to,next;
db w;
}c[E];
int head[P],t;
inline void add(int x,int y,db z){
c[++t].to=y,c[t].next=head[x],head[x]=t,c[t].w=z;
}
db dis[P];
bool in[P];
int n,m,sum;
int val[N][N],cost1[N][N],cost2[N][N];
int q[P],front,back;
int cnt[P];
inline bool spfa(int s){
q[back++]=s,in[s]=true;
if(back==P)back=;
while(front!=back){
int x=q[front++];
in[x]=false;
if(front==P)front=;
for(int i=head[x];i;i=c[i].next)
if(dis[x]+c[i].w-dis[c[i].to]>eps){
dis[c[i].to]=dis[x]+c[i].w;
cnt[c[i].to]=cnt[x]+;
if(cnt[c[i].to]==sum)return true;
if(in[c[i].to]==false){
q[back++]=c[i].to;
in[c[i].to]=true;
if(back==P)back=;
}
}
}
return false;
}
inline bool check(db mid){
memset(head,,sizeof(head)),t=;
memset(in,,sizeof(in));
memset(cnt,,sizeof(cnt));
for(int i=;i<=sum;++i)dis[i]=oo;
for(int i=;i<=n+;++i)
for(int j=;j<=m+;++j){
if(j!=m+)
add(pos(i,j),pos(i,j+),-cost1[i][j]*mid+val[i][j]);
if(j!=)
add(pos(i,j),pos(i,j-),-cost1[i][j-]*mid-val[i][j-]);
if(i!=n+)
add(pos(i,j),pos(i+,j),-cost2[i][j]*mid);
if(i!=)
add(pos(i,j),pos(i-,j),-cost2[i-][j]*mid);
}
dis[sum/]=.;
return spfa(sum/);
}
int main(){
scanf("%d%d",&n,&m);
sum=(n+)*(m+);
for(int i=;i<=n;++i)
for(int j=;j<=m;++j)
scanf("%d",&val[i][j]);
for(int i=;i<=m;++i)
for(int j=n;j>;--j)
val[j][i]+=val[j+][i];
for(int i=;i<=n+;++i)
for(int j=;j<=m;++j)
scanf("%d",&cost1[i][j]);
for(int i=;i<=n;++i)
for(int j=;j<=m+;++j)
scanf("%d",&cost2[i][j]);
db l=.,r=.,mid,ans=.;
while(l+ans_eps<r){
mid=(l+r)*0.5;
if(check(mid))
ans=mid,l=mid;
else
r=mid;
}
printf("%.3f",(double)ans);
return ;
}
【BZOJ 3232】圈地游戏 二分+SPFA判环/最小割经典模型的更多相关文章
- BZOJ 3232: 圈地游戏 分数规划+判负环
3232: 圈地游戏 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 966 Solved: 466[Submit][Status][Discuss] ...
- bzoj 3232: 圈地游戏
bzoj 3232: 圈地游戏 01分数规划,就是你要最大化\(\frac{\sum A}{\sum B}\),就二分这个值,\(\frac{\sum A}{\sum B} \geq mid\) \( ...
- bzoj 3232 圈地游戏——0/1分数规划(或网络流)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3232 当然是0/1分数规划.但加的东西和减的东西不在一起,怎么办? 考虑把它们合在一起.因为 ...
- bzoj 3232 圈地游戏 —— 01分数规划+最小割建图(最大权闭合子图)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3232 心烦意乱的时候调这道题真是...越调越气,就这样过了一晚上... 今天再认真看看,找出 ...
- bzoj 3232: 圈地游戏【分数规划+最小割】
数组开小导致TTTTTLE-- 是分数规划,设sm为所有格子价值和,二分出mid之后,用最小割来判断,也就是判断sm-dinic()>=0 这个最小割比较像最大权闭合子图,建图是s像所有点连流量 ...
- bzoj 3232: 圈地游戏 01分数规划
题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=3232 题解: 首先我们看到这道题让我们最优化一个分式. 所以我们应该自然而然地想到01分 ...
- bzoj 2132 圈地计划(黑白染色,最小割)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2132 [题意] 给定n*m个区域,建工业区价值A,建商业区价值B,如果(i,j)有k个 ...
- poj 2049(二分+spfa判负环)
poj 2049(二分+spfa判负环) 给你一堆字符串,若字符串x的后两个字符和y的前两个字符相连,那么x可向y连边.问字符串环的平均最小值是多少.1 ≤ n ≤ 100000,有多组数据. 首先根 ...
- 2018.09.12 poj3621Sightseeing Cows(01分数规划+spfa判环)
传送门 01分数规划板题啊. 发现就是一个最优比率环. 这个直接二分+spfa判负环就行了. 代码: #include<iostream> #include<cstdio> # ...
随机推荐
- 第四课:PHP 变量
变量指程序中使用的数值是可以变化的量,与常量(一旦被定义,就无法改变)相反. 变量是用于存储信息的"容器": 实例 <?php $x=5; $y=6; $z=$x+$y; e ...
- [转]Nginx伪静态配置和常用Rewrite伪静态规则集锦
Nginx伪静态配置和常用Rewrite伪静态规则集锦 作者: 字体:[增加 减小] 类型:转载 时间:2014-06-10 我要评论 伪静态是一种可以把文件后缀改成任何可能的一种方法,如果我想把ph ...
- laravel 中出现SQLSTATE[HY000] [2002] 如何解决?
在日常开发中总是难免遇到各式各样的错误,还有许多错误常常是重复出现的 以下是报错信息! SQLSTATE[HY000] [2002] ������ӷ���һ��ʱ���û���ȷ�
- web学习第二天
今天是学习web的第二天,早上用css3做了个会动的小熊,border-radius为圆的半径, .smallxiong { width: 400px; height: 400px; ...
- A*与IDA*的奇妙之旅
因为A*卡了一天QAQ 那么,A*是什么呢? A* A*是对于bfs的优化,启发式搜索. 例如下图: 不错,在这张图上,小人去找电脑,用bfs的话: 黄色就是bfs的搜索范围...不要问我为什么选黄色 ...
- 类 java.util.Collections 提供了对Set、List、Map进行排序、填充、查找元素的辅助方法。
类 java.util.Collections 提供了对Set.List.Map进行排序.填充.查找元素的辅助方法. 1. void sort(List) //对List容器内的元素排序,排序的规 ...
- MYSQL--事务处理(转)
事务处理在各种管理系统中都有着广泛的应用,比如人员管理系统,很多同步数据库操作大都需要用到事务处理.比如说,在人员管理系统中,你删除一个人员,你即需要删除人员的基本资料,也要删除和该人员相关的信息,如 ...
- Codeforces Round #345 Div.1 D.Zip-line 动态最长上升子序列
题意概述: 给出一个长度为N的序列和M组询问,问假设把某个位置的值改成另一个给出的值之后,序列的最长上升子序列的长度. N,M<=400000. 分析: 考虑某个位置的值改动后这个位置和最长上升 ...
- markdown(自己看)
https://www.cnblogs.com/james-lee/p/6847906.html https://maxiang.io/
- EasyUI 显示表单数据 小记
界面图: