【JZOJ3215】【SDOI2013】费用流
╰( ̄▽ ̄)╭
对于一张给定的 运输网络 ,Alice 先确定一个最大流 ,如果有多种解, Alice 可以任选一种; 之后 Bob在每条边上分配单位花费 (单位花费必须是非负实数), 要求所有边的单位花费之和等于 P。总费用等于每一条边 的实际流量乘以该边的单位花费。 需要注意到, Bob在分配单位花费之前,已经知道Alice 所给出的最大流方案。
现在 Alice 希望总费用尽量小,而Bob希望总费用尽量大。我们想知道, 如 果两个人都执行最优策略 ,最大流的值和总费用分别为多少。
对于 100% 的测试数据: N≤100 ,M≤1000 。
对于 100% 的测试数据: 所有点的编号在 1..N 范围内。 1≤每条边 的最大 流 量≤50000 。1≤P≤10 。给定运输网络中不会有起点和 终点 相同的边。
(⊙ ▽ ⊙)
显然Bob要把所有费用全部分配给实际流量最大的边。
所以Alice在满足最大流最大之余,使得流量最大的边最小。
所以二分后再用最大流判断就可以了。
( ̄~ ̄)
#include<iostream>
#include<algorithm>
#include<stdio.h>
#include<math.h>
#include<string.h>
#define ll long long
#define eps 10e-7
using namespace std;
const char* fin="ex3215.in";
const char* fout="ex3215.out";
const int inf=0x7fffffff;
const int maxn=2007;
int n,m,n1,i,j,k;
int a[maxn][2],fi[maxn],ne[maxn],la[maxn],tot;
double l,r,mid,va[maxn],b[maxn],Ans;
int bz[maxn],card[maxn];
double Abs(double a){
return a>0?a:-a;
}
void add_line(int a,int b,double c){
tot++;
ne[tot]=fi[a];
la[tot]=b;
va[tot]=c;
fi[a]=tot;
}
void add(int a,int b,double c){
add_line(a,b,c);
add_line(b,a,0);
}
double gap(int v,double flow){
int i,k;
double use=0,j;
if (v==n) return flow;
for (k=fi[v];k;k=ne[k])
if (bz[la[k]]+1==bz[v] && Abs(va[k])>eps){
j=gap(la[k],min(va[k],flow-use));
use+=j;
va[k]-=j;
va[k^1]+=j;
if (Abs(flow-use)<eps || bz[1]==n) return use;
}
if (!--card[bz[v]]) bz[1]=n;
card[++bz[v]]++;
return use;
}
double flow(){
double ans=0;
memset(card,0,sizeof(card));
memset(bz,0,sizeof(bz));
card[0]=n;
while (bz[1]<n){
ans+=gap(1,inf);
}
return ans;
}
bool judge(double MAX){
int i,j,k;
memset(fi,0,sizeof(fi));
tot=1;
for (i=1;i<=m;i++) add(a[i][0],a[i][1],min(MAX,b[i]));
return Abs(flow()-Ans)<eps;
}
int main(){
scanf("%d%d%d",&n,&m,&n1);
tot=1;
for (i=1;i<=m;i++) scanf("%d%d%lf",&a[i][0],&a[i][1],&b[i]),add(a[i][0],a[i][1],b[i]);
Ans=flow();
l=0;
r=50000;
while (r-l>eps){
mid=(l+r)/2;
if (judge(mid)) r=mid;
else l=mid;
}
printf("%d\n%.4lf",int(Ans+eps),l*n1 );
return 0;
}
(⊙v⊙)
要注意的是网络流的实现时的问题:
double gap(int v,double flow){
int i,k;
double use=0,j;
if (v==n) return flow;
for (k=fi[v];k;k=ne[k])
if (bz[la[k]]+1==bz[v] && Abs(va[k])>eps){
j=gap(la[k],min(va[k],flow-use));
use+=j;
va[k]-=j;
va[k^1]+=j;
if (Abs(flow-use)<eps || bz[1]==n) return use;
}
if (!--card[bz[v]]) bz[1]=n;
card[++bz[v]]++;
return use;
}
1.三个中两个return返回的都是use;
2.当use==flow使,直接返回use;
3.到达汇点,返回flow。
【JZOJ3215】【SDOI2013】费用流的更多相关文章
- BZOJ3130: [Sdoi2013]费用流[最大流 实数二分]
3130: [Sdoi2013]费用流 Time Limit: 10 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 960 Solved: 5 ...
- bzoj千题计划133:bzoj3130: [Sdoi2013]费用流
http://www.lydsy.com/JudgeOnline/problem.php?id=3130 第一问就是个最大流 第二问: Bob希望总费用尽量大,那肯定是把所有的花费加到流量最大的那一条 ...
- P3305 [SDOI2013]费用流
题目描述 Alice和Bob在图论课程上学习了最大流和最小费用最大流的相关知识. 最大流问题:给定一张有向图表示运输网络,一个源点S和一个汇点T,每条边都有最大流量. 一个合法的网络流方案必须满足: ...
- luogu P3305 [SDOI2013]费用流
题目链接 bz似乎挂了... luogu P3305 [SDOI2013]费用流 题解 dalao告诉我,这题 似乎很水.... 懂了题目大意就可以随便切了 问1,最大流 问2,二分最大边权求,che ...
- BZOJ 3130: [Sdoi2013]费用流 网络流+二分
3130: [Sdoi2013]费用流 Time Limit: 10 Sec Memory Limit: 128 MBSec Special JudgeSubmit: 1230 Solved: ...
- BZOJ 3130: [Sdoi2013]费用流 网络流 二分 最大流
https://www.lydsy.com/JudgeOnline/problem.php?id=3130 本来找费用流的题,权当复习一下网络流好了. 有点麻烦的是double,干脆判断大小或者二分增 ...
- BZOJ3130 [Sdoi2013]费用流
AC通道:http://www.lydsy.com/JudgeOnline/problem.php?id=3130 这题codevs上也有,不过数据挂了[要A得看discuss]. 题目大意: Ali ...
- bzoj 3130 [Sdoi2013]费用流(二分,最大流)
Description Alice和Bob在图论课程上学习了最大流和最小费用最大流的相关知识. 最大流问题:给定一张有向图表示运输网络,一个源点S和一个汇点T,每条边都有最大流量.一个合法的网络 ...
- [SDOI2013]费用流
然而这是一道网络流... 如果满足Bob,使总费用最大: 设最大流的每条边流量(不是容量)为w[i],分配到每条边的费用为p[i],最大流量为wmax,p[i]的和为P 那么显然w[i] * p[i] ...
- Luogu P3305 [SDOI2013]费用流 二分 网络流
题目链接 \(Click\) \(Here\) 非常有趣的一个题目. 关键结论:所有的单位费用应该被分配在流量最大的边上. 即:在保证最大流的前提下,使最大流量最小.这里我们采用二分的方法,每次判断让 ...
随机推荐
- badboy的录制和jmeter的使用
v Jmeter是什么 Apache Jmeter是Apache组织开发的基于Java的压力测试工具. Jmeter可以用于对服务器.网络或对象模拟巨大的负载,来自不同压力类别下测试它们的强度和分析 ...
- 手机端META详细解释
一.天猫 <title>天猫触屏版</title> <meta content="text/html; charset=utf-8" http-equ ...
- OSG实现正八面体剖分成球
#include<Windows.h> #include<osg/Node> #include<osg/Geode> #include<osg/Group&g ...
- Leetcode951. Flip Equivalent Binary Trees翻转等价二叉树
我们可以为二叉树 T 定义一个翻转操作,如下所示:选择任意节点,然后交换它的左子树和右子树. 只要经过一定次数的翻转操作后,能使 X 等于 Y,我们就称二叉树 X 翻转等价于二叉树 Y. 编写一个判断 ...
- Ubuntu 14.04 Ruby 2.3.3 安装
在Ubuntu 14.04通过下载Ruby源码包进行安装. 第一步,更新apt-get sudo apt-get update 通过apt-get安装ruby依赖 sudo apt-get insta ...
- set_clock_latency
set_clock_latancy用于定于虚拟时钟与真实时钟的延时 考虑最糟糕的情况,评估setup时数据会使用最大延时,时钟使用最小延时:评估hold时,数据使用最小延时,时钟使用最大延时.
- JS--封装JS跳转页面函数
//JS跳转页面 function gourl($iAlert,$iPage,$history='',$target="window") { if ($iAlert != &quo ...
- SQL有意思的面试题
1.中软国际 SQL行转列 变成 --数据准备create table t_test( year int, month int, sale int, primary key (year, mon ...
- 在Bat批处理中调用Powershell脚本
##如何在BAT中调用powershell,把下面代码另存为bat格式pushd %~dp0powershell.exe -command ^ "& {set-executionp ...
- 原生JS实现简易计算器
<!doctype html> <html> <head> <meta charset="utf-8"> <title> ...