洛谷2494 [SDOI2011]保密 (分数规划+最小割)
自闭一早上
分数规划竟然还能被卡精度
首先假设我们已经知道了到每个出入口的时间(代价)
那我们应该怎么算最小的和呢?
一个比较巧妙的想法是,由于题目规定的是二分图。
我们不妨通过最小割的形式。
表示这个基地必须从两个口之一进,从\(S\)连到奇数点,偶数点连到\(T\),流量是到这个点的时间。
然后对于每个空腔的\(u和v,(u->v,inf)\)表示这个二者至少要到一个。
那么这样跑一遍最小割,就表示经过所有空腔的最小代价。
那么现在其实问题就转化成了
如果求到一个点的时间\(a[i]\)
观察题目中的柿子。
\(a[i]=\frac{\sum t}{\sum s}\)
哎!这个是不是可以直接分数规划啊。
因为有无限的军队,所以每个点之间是可以单独处理的。
首先我们先考虑对于任意一个点\(x\)
不妨直接二分
设\(mid>\frac{\sum t}{\sum s}\)
则
\]
如果我们将边权设为\(t-mid*s\)
那么我们只需要判断到一个点的最短路是否是小于0,如果小于,那么\(ans<mid\),继续二分即可。
那么我们对于每一个点都做一个这样的过程,\(a[i]\)也就能求出来了啦
qwq
这有一个需要注意的地方,也是我做的时候看题解才用的一个剪枝。
就是我们用\(spfa\)求最短路的时候,只要遇到了目标点,并且\(dis<=0\),直接\(return\)
这样能大大加快你程序的速度。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define mk make_pair
#define ll long long
#define db double
using namespace std;
inline int read()
{
int x=0,f=1;char ch=getchar();
while (!isdigit(ch)) {if (ch=='-') f=-1;ch=getchar();}
while (isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
}
const int maxn = 1410;
const int maxm = 1e6+1e2;
const double inf = 1e12;
const double eps = 1e-4 ;
int point[maxn],nxt[maxm],to[maxm];
db val[maxm];
int cnt=1,n,m;
int h[maxn];
int x[maxm],y[maxm];
double w[maxm],p[maxm];
int s,t;
int st;
double dis[maxn],cost[maxm];
int vis[maxn];
void init()
{
cnt=1;
memset(point,0,sizeof(point));
}
void addedge(int x,int y,db w)
{
nxt[++cnt]=point[x];
to[cnt]=y;
point[x]=cnt;
val[cnt]=w;
}
void add(int x,int y,double w)
{
//printf("%d %d %.2lf\n",x,y,w);
nxt[++cnt]=point[x];
to[cnt]=y;
cost[cnt]=w;
point[x]=cnt;
}
void insert(int x,int y,db w)
{
//cout<<x<<" "<<y<<" "<<w<<endl;
addedge(x,y,w);
addedge(y,x,0);
}
queue<int> q;
void spfa(int s,int ed)
{
for (int i=1;i<=maxn-2;i++) dis[i]=inf;
memset(vis,0,sizeof(vis));
while (!q.empty()) q.pop();
dis[s]=0;
q.push(s);
while (!q.empty())
{
int x = q.front();
q.pop();
vis[x]=0;
for (int i=point[x];i;i=nxt[i])
{
int p = to[i];
if (dis[p]>dis[x]+cost[i])
{
dis[p]=dis[x]+cost[i];
if (p==ed && dis[p]<=-eps) return;
if (!vis[p])
{
q.push(p);
vis[p]=1;
}
}
}
}
}
bool bfs(int s)
{
memset(h,-1,sizeof(h));
while (!q.empty()) q.pop();
h[s]=0;
q.push(s);
while (!q.empty())
{
int x = q.front();
q.pop();
for (int i=point[x];i;i=nxt[i])
{
int p = to[i];
if (h[p]==-1 && val[i]>0)
{
h[p]=h[x]+1;
q.push(p);
}
}
}
if (h[t]==-1) return false;
return true;
}
db dfs(int x,db low)
{
if (x==t ||low==0) return low;
db totflow=0;
for (int i=point[x];i;i=nxt[i])
{
int p = to[i];
if (val[i]>0 && h[p]==h[x]+1)
{
db tmp = dfs(p,min(low,val[i]));
val[i]-=tmp;
val[i^1]+=tmp;
low-=tmp;
totflow+=tmp;
if (low==0) return totflow;
}
}
if (low>0) h[x]=-1;
return totflow;
}
db dinic()
{
db ans=0;
while (bfs(s))
{
ans=ans+dfs(s,inf);
}
return ans;
}
double uu=0;
bool check(double mid,int xx)
{
init();
for (int i=1;i<=m;i++)
add(x[i],y[i],w[i]-mid*p[i]);
spfa(n,xx);
return (dis[xx]<=-eps);
}
double solve(int x)
{
double l=0,r=2e9;
double ans=inf;
while (r-l>=1e-4)
{
double mid = (l+r)/2;
if(check(mid,x)) r=mid,ans=mid;
else l=mid;
}
return ans;
}
double a[maxn];
signed main()
{
n=read(),m=read();
for (int i=1;i<=m;i++)
scanf("%lld%lld%lf%lf",&x[i],&y[i],&w[i],&p[i]);
int num1=read(),num2=read();
init();
s=maxn-10;
t=s+1;
for (int i=1;i<=num2;i++)
{
double pp = solve(i);
a[i]=pp;
}
init();
for (int i=1;i<=num2;i++)
{
if (i&1) insert(s,i,a[i]);
else insert(i,t,a[i]);
}
for (int i=1;i<=num1;i++)
{
int u=read(),v=read();
if (v&1) swap(u,v);
if (a[u]==inf && a[v]==inf)
{
cout<<-1;
return 0;
}
insert(u,v,inf);
}
double ptx = dinic();
printf("%.1lf\n",ptx);
return 0;
}
洛谷2494 [SDOI2011]保密 (分数规划+最小割)的更多相关文章
- 【洛谷P2494】 [SDOI2011]保密(分数规划+最小割)
洛谷 题意: 题意好绕好绕...不想写了. 思路: 首先类似于分数规划做法,二分答案得到到每个点的最小危险度. 然后就是在一个二分图中,两边撤掉最少的点(相应代价为上面算出的危险度)及相应边,使得中间 ...
- zoj 2676 Network Wars 0-1分数规划+最小割
题目详解出自 论文 Amber-最小割模型在信息学竞赛中的应用 题目大意: 给出一个带权无向图 G = (V,E), 每条边 e属于E都有一个权值We,求一个割边集C,使得该割边集的平均边权最小,即最 ...
- 【BZOJ3232】圈地游戏 分数规划+最小割
[BZOJ3232]圈地游戏 Description DZY家的后院有一块地,由N行M列的方格组成,格子内种的菜有一定的价值,并且每一条单位长度的格线有一定的费用. DZY喜欢在地里散步.他总是从任意 ...
- bzoj 3232: 圈地游戏【分数规划+最小割】
数组开小导致TTTTTLE-- 是分数规划,设sm为所有格子价值和,二分出mid之后,用最小割来判断,也就是判断sm-dinic()>=0 这个最小割比较像最大权闭合子图,建图是s像所有点连流量 ...
- bzoj 3232 圈地游戏 —— 01分数规划+最小割建图(最大权闭合子图)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3232 心烦意乱的时候调这道题真是...越调越气,就这样过了一晚上... 今天再认真看看,找出 ...
- 【洛谷P3973】[TJOI2015]线性代数(最小割)
洛谷 题意: 给出一个\(n*n\)的矩阵\(B\),再给出一个\(1*n\)的矩阵\(C\). 求一个\(1*n\)的\(01\)矩阵\(A\),使得\(D=(A\cdot B-C)\cdot A^ ...
- HDU 2676 Network Wars 01分数规划,最小割 难度:4
http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=1676 对顶点i,j,起点s=1,终点t=n,可以认为题意要求一组01矩阵use ...
- 【洛谷 P3227】 [HNOI2013]切糕(最小割)
题目链接 每层每个位置向下一层这个位置连边,流量为下一层这个位置的\(f\),源点向第一层连,流量第一层每个位置的费用,最后一层向汇点连,流量\(INF\). 这样就得到了\(P*Q\)条链,不考虑\ ...
- 洛谷P3355 骑士共存问题(最小割)
传送门 de了两个小时的bug愣是没发现错在哪里……没办法只好重打了一遍竟然1A……我有点想从这里跳下去了…… 和方格取数问题差不多,把格子按行数和列数之和的奇偶性分为黑的和白的,可以发现某种颜色一定 ...
随机推荐
- react的每个方法为什么一定要bind this
一开始学习react的时候就了解了react的每个方法都要bind(this)或者使用箭头函数绑定this的指向,到底是为什么要这么写呢,当时要学习的东西太多了就没在意,今天特别好奇(不搞懂不吃饭的态 ...
- 关于notepad++使用的那些事儿
时间:2019-04-11 整理:PangYuaner 标题:Notepad++正则表达式语法 地址:https://www.cnblogs.com/kekec/p/5255475.html 标题:N ...
- vue 导出excel后端返回乱码下载不了的解析问题
有俩种方法可以用 第一种方法: 解析:直接解析并且下载后端的乱码 this.download('后端给的导出excel的方法', { ...this.queryParams }, ` ...
- 2021.9.12周六PAT甲级考试复盘与总结
周六PAT甲级考试复盘与总结 先说结论:仍未步入"高手"行列:现在的学习节奏与方法是对的,有十万分的必要坚持下去. 题目 知识点 分数 T1 前缀和.二分 11 / 20 T2 排 ...
- Vue组件传值(二)之 非父子组件传值
Vue中非父子组件之间是如何实现通信的? 本章主要讲的是非父子组件传值,父子组件传值请看上一篇文章. 1.创建新的Vue实例引入项目中,通过$emit.$on来实现非父子组件传值: 1 <!DO ...
- Linux安装Cockpit监控服务
CentOS/RHEL 8的新特性之一就是自带了一个cockpit的监控服务.通过c/s架构模式运行,客户端输入ip:端口即可访问 这类似于glances监控. 如果你不是使用的centos/rhel ...
- 学习PHP中的信息格式化操作
在国际化组件的学习过程中,我们已经接触过了 NumberFormatter 这种数字的格式化操作,它可以让我们将数字转换成标准格式.货币.本地语言等形式.今天我们来学习的是另一种专门用于信息格式化的类 ...
- PHP中的MySQLi扩展学习(五)MySQLI_STMT对象操作
就像 PDO 中的 PDO_Statment 对象一样,MySQLI_STMT 对象也是一个预处理语句所形成的对象,专门用来操作 MySQLi 所生成的预处理语句的.其实操作方式之类也都比较相似,不外 ...
- php move_uploaded_file保存文件失败
move_uploaded_file保存失败后找错,先使用了try catch,但是没输出信息,才知道该函数在php中是警告属于error,不属于exeption,因此不能通过简单的if(!...)处 ...
- Java之SpringBoot自定义配置与整合Druid
Java之SpringBoot自定义配置与整合Druid SpringBoot配置文件 优先级 前面SpringBoot基础有提到,关于SpringBoot配置文件可以是properties或者是ya ...