Coding Contest HDU - 5988

题意:

有n个点,m个边,每个点有人数和食物数,每个人都要吃一份食物,如果该点的食物不够,他们就要去其他点,每个边最多只能走c次,每次有人走一条路,这条路就有p的概率坏掉。第一个人通过时不会坏掉。求最小破坏的电线的概率

题解:

不难看出是一个网络流,但是不知道该怎么建边(这也是网络流最难的部分)
参考题解
每条边都有走的次数(当作流量),每个边走一次发生破坏的概率为p(流量1,费用p),我们开始建立费用流图。根据题意每个边坏掉概率,如果走多个边那概率应该相乘,但是费用流往往是累加的,如何将相乘转成累加?我们可以通过对每个概率取log当成费用,在log下所有都是相加减。
但是题目的概率都是小于1的,如果取log都是负数,费用为负,这跑出来有问题(跑出来的费用会朝着更小走)。如何解决?那么取个负数呢,还是不行,因为取负后最小的变成最大的,跑出来就成最大费用了
此时我们应该这样考虑,题目要求求最小概率,也就是1-最大概率,因此我们把每条边的概率赋值为1-p,然后取反取log,这样跑正好得到的是最小费用,取出来之后再用1减去就好了
add(u,v,f,-log2(1-p)),f为容量,p为概率,从u到v的边
其他如何建边:
建立源点s,汇点t,对于S>B(人多),从源点连一条流量为S[i]-B[i],费用为0,对于s<b(粮食多)的,从该点向t连个边,费用为B[i]-S[i]
题目还说第一次踩不会坏,所以从原有的边取一条出来,流量为1,费用为0

代码:

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<iostream>
#include<cmath>
#include<queue>
#include<string>
#include<functional>
typedef long long LL;
using namespace std;
#define MAXN 110
#define MAXM 25000
#define ll l,mid,now<<1
#define rr mid+1,r,now<<1|1
#define lson l1,mid,l2,r2,now<<1
#define rson mid+1,r1,l2,r2,now<<1|1
#define pi acos(-1.0)
#define INF 2e9
const double eps = 1e-8;
const int mod = 1e9 + 7;
struct Edge
{int to, next, cap, flow;double cost;
}edge[MAXM];
int head[MAXN], tol;
int pre[MAXN];
double dis[MAXN];
bool vis[MAXN];
int N;//节点总个数,节点编号从0~N-1
void init(int n)
{N = n;tol = 0;memset(head, -1, sizeof(head));
}
void addedge(int u, int v, int cap, double cost)
{edge[tol].to = v;edge[tol].cap = cap;edge[tol].cost = cost;edge[tol].flow = 0;edge[tol].next = head[u];head[u] = tol++;edge[tol].to = u;edge[tol].cap = 0;edge[tol].cost = -cost;edge[tol].flow = 0;edge[tol].next = head[v];head[v] = tol++;
}
bool spfa(int s, int t)
{queue<int>q;for (int i = 0; i <= N; i++){dis[i] = INF;vis[i] = false;pre[i] = -1;}dis[s] = 0;vis[s] = true;q.push(s);while (!q.empty()){//cout<<1<<endl;  int u = q.front();q.pop();vis[u] = false;for (int i = head[u]; i != -1; i = edge[i].next){int v = edge[i].to;if (edge[i].cap > edge[i].flow &&dis[v]-dis[u]-edge[i].cost>eps){dis[v] = dis[u] + edge[i].cost;pre[v] = i;if (!vis[v]){vis[v] = true;q.push(v);}}}}if (pre[t] == -1)return false;else return true;
}
//返回的是最大流,cost存的是最小费用
int minCostMaxflow(int s, int t, double &cost)
{int flow = 0;cost = 0;while (spfa(s, t)){//cout<<1<<endl;  int Min = INF;for (int i = pre[t]; i != -1; i = pre[edge[i ^ 1].to]){if (Min > edge[i].cap - edge[i].flow)Min = edge[i].cap - edge[i].flow;}for (int i = pre[t]; i != -1; i = pre[edge[i ^ 1].to]){edge[i].flow += Min;edge[i ^ 1].flow -= Min;cost += edge[i].cost * Min;}flow += Min;}return flow;
}
int main()
{int t;scanf("%d", &t);while (t--){int n, m;scanf("%d%d", &n, &m);init(n + 1);for (int i = 1; i <= n; i++){int s, b;scanf("%d%d", &s, &b);int f = s - b;//0是源点,n+1是汇点 if (f > 0)///如果人多addedge(0, i, f, 0);else if (f < 0)///如果面包多addedge(i, n + 1, -f, 0);}while (m--){int u, v, f;double w;scanf("%d%d%d%lf", &u, &v, &f, &w);///f是这条路的容量w = -log2(1 - w);///这样就是正值了if (f > 0)addedge(u, v, 1, 0);///第一个人经过时,不破坏if (f - 1>0)addedge(u, v, f - 1, w);///第大于等于2个人经过时破坏}double cost = 0;minCostMaxflow(0, n + 1, cost);cost = 1 - pow(2, -cost);printf("%.2lf\n", cost);}
}

Coding Contest HDU - 5988相关推荐

  1. 【HDU - 5988】Coding Contest(网络流费用流,改模板)

    题干: A coding contest will be held in this university, in a huge playground. The whole playground wou ...

  2. HDU - 5988 Coding Contest(最大费用最大流+思维建边)

    题目链接:点击查看 题目大意:给出n个点和m条边,每个点有ai个人和bi份食物,每条边最多能通过ci个人,以及除了第一个人之外,其他人通过每条边触电的概率为pi,在以上约束下,要求每个人都必须吃到一份 ...

  3. HDU 5988 最小费用流

    链接: http://acm.split.hdu.edu.cn/showproblem.php?pid=5988 题意: n个点,每个点有a个人和b包饭,m条边,第一次经过这条边没有问题,以后每一个经 ...

  4. 【最小费用最大流(改进Dijkstra)】2016 icpc 青岛 G - Coding Contest

    题目:https://vjudge.net/contest/412116#problem/G 题意:t组样例,n个点,每个点有sis_isi​个人和bib_ibi​份物资.m条边,每条边从第二次开始, ...

  5. Helvetic Coding Contest 2019 online mirror (teams allowed, unrated)

    http://codeforces.com/contest/1184 A1 找一对整数,使x^x+2xy+x+1=r 变换成一个分式,保证整除 #include<iostream> #in ...

  6. Helvetic Coding Contest 2018 online mirror (teams allowed, unrated) C2.Encryption (medium)

    传送门:Encryption(medium) 题意:N个数,分成k个连续的序列,把每个序列的数累加取模p,求k个取模结果累加的最大值. 分析:DP, 复杂度O(Nkp) 设dp[i][j]为取模结果i ...

  7. CF 690C3. Brain Network (hard) from Helvetic Coding Contest 2016 online mirror (teams, unrated)

    题目描述 Brain Network (hard) 这个问题就是给出一个不断加边的树,保证每一次加边之后都只有一个连通块(每一次连的点都是之前出现过的),问每一次加边之后树的直径. 算法 每一次增加一 ...

  8. oracle导入提示字符过长,Oracle 解决【ORA-01704:字符串文字太长】

    错误提示:oracle在toad中执行一段sql语句时,出现错误'ORA-01704:字符串文字太长'.如下图: 原因:一般为包含有对CLOB字段的数据操作.如果CLOB字段的内容非常大的时候,会导致 ...

  9. bitset HDU6515 Coding Problem

    Coding Problem [ HDU - 6515 ] 题目大意:给你一个字符串,每个字母的ASCII二级制颠倒过来组成一个01数组. 然后这个数组每六位组成一个数字的ASCII输出 一道模拟题, ...

最新文章

  1. Properties持久的属性集
  2. C# Obsolete(已弃用方法属性)
  3. Linux下eclipse及mysql安装,c++访问mysql数据库
  4. Eclipse的Ctrl+鼠标左键的智能化链接配置
  5. 952计算机网络是那本书,952计算机网络复习参考提纲.doc
  6. 发现几个常用的asp.net MVC Helper 源码
  7. 【剑指offer】面试题62:圆圈中最后剩下的数字(Java)
  8. 两千块钱带来的 quot;希望quot;
  9. CSDN markdown编辑器 页面内跳转目录
  10. is_numeric 检测变量是否为数字或数字字符串
  11. opencv中滚动条操作
  12. matlab时空地理回归,★时空地理加权回归space-time GWR GTWR分析软件-香港中文大学-黄波教授...
  13. ASCII 码表(完整版)
  14. 项目管理工具与技术---项目风险管理中的工具与技术
  15. 华为认证云服务工程师(HCIA-Cloud ServiceV3.0)-- 认证介绍
  16. 全量发布与灰度发布(APP版本发布)
  17. 科学论文类型分类(letter,nbsp;p…
  18. 工程量计算稿1.55安装步骤 v1.55pjb
  19. set feedback on
  20. c语言中03怎么表示成30,C语言入门篇-03

热门文章

  1. 理工男都能有多痴情?
  2. 全景图解高铁数据,谁是最有潜力的高铁城市?
  3. SparkSQL DataFrame进阶篇
  4. mysql左键关联_MySQL的:左键删除重复列加入,3个表
  5. 纳尼???我JVM优化过头了,直接把异常信息优化没了?怎么办
  6. 大文本导入mysql_大文本数据,导入导出到数据库
  7. 360浏览器5兼容模式吗_个独模式真能将企业总体税负降低至5%吗,为何大家都热衷核定征收...
  8. linux java升级版本_为嵌入式Linux设备实现更新/升级系统
  9. leetcode47. 全排列 II
  10. 哈希表(散列表)基础概念与经典题目(Leetcode题解-Python语言)之上——原理与设计