[Luogu 1730]最小密度路径
Description
给出一张有N个点M条边的加权有向无环图,接下来有Q个询问,每个询问包括2个节点X和Y,要求算出从X到Y的一条路径,使得密度最小(密度的定义为,路径上边的权值和除以边的数量)。
Input
第一行包括2个整数N和M。
以下M行,每行三个数字A、B、W,表示从A到B有一条权值为W的有向边。
再下一行有一个整数Q。
以下Q行,每行一个询问X和Y,如题意所诉。
Output
对于每个询问输出一行,表示该询问的最小密度路径的密度(保留3位小数),如果不存在这么一条路径输出“OMG!”(不含引号)。
Sample Input
3 3
1 3 5
2 1 6
2 3 6
2
1 3
2 3
Sample Output
5.000
5.500
Hint
1 ≤ N ≤ 50,1 ≤ M ≤ 1000,1 ≤ W ≤ 100000,1 ≤ Q ≤ 100000
题解
考虑转化成我们熟悉的问题解决。
由于都是求最小,很容易想到和此题类似的一个问题,求任两点间的最短路,能否借鉴 $Floyd$ 算法来解决呢?
本题不同点在于,还要除以一个边数。因为这个除法的缘故,使得 $Floyd$ 算法的最优子结构性质被破坏,假设存在路径 $i -> k -> j$,它的最小密度路径并不一定是 $i -> k$ 的最小密度路径加上 $k -> j$ 的最小密度路径。
例:设$[A, B]$表示路径的权值和为 $A$,通过了 $B$ 条边。假设从 $i -> k$ 存在着两条路径 $L1[2, 3]$以及 $L2[8, 10]$,从 $k -> j$ 存在着两条路径 $L3[1, 2]$以及 $L4[51, 100]$,很明显 $i -> k$ 的最小密度路径是 $L1$,$k -> j$ 的最小密度路径是 $L3$,但是 $i -> k -> j$ 的最小密度路径却是 $L1 + L4$。
有否办法去掉这个除法的影响?
回到问题特性,是有向无环图,一条路径最多只能经过 $N-1$条边,于是我们可以对边数进行枚举,即把答案的分母枚举了,剩下的就是让答案的分子最小化(答案是 权值和/边数),这就回到我们熟悉的问题:求最短。
在 $Floyd$ 的基础上重新划分阶段定义状态:
第 $k$ 个阶段表示恰好通过 $k$ 条边两点间的最短路,这样的话最优子结构以及无后效性都满足($k$ 的阶段的最优取值一定需要靠之前阶段的最优值,当然也不可能影响到之前阶段的取值了。)
定义状态 $f(i,j,k)$表示从 $i$ 到 $j$ 恰好经过 $k$ 条边的最短路,类似$Floyd$ 的算法得出 $DP$ 方程:
$$f(i,j,k)=Min(f(i,h,g)+f(h,j,k-g))$$
这个方程是 $5$ 维的,会超时,如何减小维数呢?
考虑在何处重复决策。注意到 $f(i,j,k)$的选择路径 $V1-V2-...-Vk$,实际上我们只要找到这里的一个点决策即可,而不需每个点都判断过去。这样就很容易想到在最后一个点进行决策。
$f(i,j,k)=Min(f(i,h,k-1)+f(h,j,1))$。数据范围不大,询问比较多,考虑用 $dp$ 直接算出所有点对的答案.因为密度 =$val \over R$ 所以考虑 $f[x][y][R]$ 为 $x=>y$ 经过 $R$ 条边的最小值 ,$ans={f[x][y][R] \over R}$
状态转移为:
$$f[i][j][R]=f[i][k][R-1]+f[k][j][1]$$
//It is made by Awson on 2017.10.30
#include <set>
#include <map>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <cstdio>
#include <string>
#include <cstdlib>
#include <cstring>
#include <iostream>
#include <algorithm>
#define LL long long
#define Max(a, b) ((a) > (b) ? (a) : (b))
#define Min(a, b) ((a) < (b) ? (a) : (b))
using namespace std;
int INF;
const double eps = 1e-; int n, m, u, v, c;
int f[][][]; void work() {
memset(f, /, sizeof(f)); INF = f[][][];
scanf("%d%d", &n, &m);
while (m--) {
scanf("%d%d%d", &u, &v, &c);
f[u][v][] = Min(c, f[u][v][]);
}
for (int l = ; l < n; l++)
for (int k = ; k <= n; k++)
for (int i = ; i <= n; i++)
if (i != k)
for (int j = ; j <= n; j++)
if (j != k && j != i)
f[i][j][l] = Min(f[i][j][l], f[i][k][l-]+f[k][j][]);
scanf("%d", &m);
while (m--) {
scanf("%d%d", &u, &v);
if (u == v) {
printf("%.3lf\n", 0.0); continue;
}
double ans = INF;
for (int i = ; i < n; i++) if (f[u][v][i] != INF) ans = min(ans, (double)f[u][v][i]/(double)i);
if (abs(ans-INF) <= eps) printf("OMG!\n");
else printf("%.3lf\n", ans);
}
}
int main() {
work();
return ;
}
[Luogu 1730]最小密度路径的更多相关文章
- Luogu P1730 最小密度路径(最短路径+dp)
P1730 最小密度路径 题面 题目描述 给出一张有 \(N\) 个点 \(M\) 条边的加权有向无环图,接下来有 \(Q\) 个询问,每个询问包括 \(2\) 个节点 \(X\) 和 \(Y\) , ...
- [洛谷P1730] 最小密度路径
类型:Floyd 传送门:>Here< 题意:定义一条路径密度 = 该路径长度 / 边数.给出一张$DAG$,现有$Q$次询问,每次给出$X,Y$,问$X,Y$的最小密度路径($N \le ...
- 【洛谷P1730】最小密度路径
题目大意:给定一个 N 个点,M 条边的有向图,现有 Q 个询问,每次询问 X 到 Y 的最小密度路径是多少.最小密度路径的定义是路径长度除以路径边数. 题解:利用矩阵乘法,可以预处理出从 X 到 Y ...
- 洛谷P1730 最小密度路径(floyd)
题意 题目链接 Sol zz floyd. 很显然的一个dp方程\(f[i][j][k][l]\)表示从\(i\)到\(j\)经过了\(k\)条边的最小权值 可以证明最优路径的长度一定\(\leqsl ...
- 洛谷P1730最小密度路径
题目传送门; 首先理解题目,究其本质就是一个最短路问题,而且数据范围贼水,用floyd完全没问题,但是题目有变化,要求出路径边权值与边数之比,这里就可以考虑在把floyd中的二维数组变为三维,f[ i ...
- Luogu 2764 最小路径覆盖问题 / Libre 6002 「网络流 24 题」最小路径覆盖 (网络流,最大流)
Luogu 2764 最小路径覆盖问题 / Libre 6002 「网络流 24 题」最小路径覆盖 (网络流,最大流) Description 给定有向图G=(V,E).设P是G的一个简单路(顶点不相 ...
- luogu P2764 最小路径覆盖问题
题目描述 给定有向图G=(V,E).设P 是G 的一个简单路(顶点不相交)的集合.如果V 中每个顶点恰好在P 的一条路上,则称P是G 的一个路径覆盖.P 中路径可以从V 的任何一个顶点开始,长度也是任 ...
- 求最大边/最小边的比值最小的路径 codevs 1001 舒适的路线
codevs 1001 舒适的路线 2006年 时间限制: 2 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description Z小镇是一个景色宜人 ...
- HDU 4862 Jump(最小K路径覆盖)
输入一个n×m网格图,每个结点的值为0-9,可以从任意点出发不超过k次,走完每个点且仅访问每个结点一次,问最终的能量最大值.不可全部走完的情况输出-1. 初始能量为0. 而结点(x,y)可以跳跃到结点 ...
随机推荐
- windows下apache报os 10048错误
在apache的bin目录下运行httpd -k install,报错os10048 (错误信息是跟443端口有关),网上的答案说的是改掉httpd.conf里的默认端口或者关闭占用端口的进程,默认端 ...
- 用Python满足满足自己的“小虚荣”
首先声明,学习这个只是为了好玩,只是为了好玩,并不是想用这个弄虚作假,做一些不好的事情!一心想做技术人,自制自治! 我们有时候发布一篇日志,或者是一篇博文,总希望自己的浏览量能高点,这样看起来也倍有面 ...
- 第二次作业之微信小程序
2.1 介绍产品相关信息 你选择的产品是? 微信小程序 为什么选择该产品作为分析? 在等待了1年多以后,小程序终于在今年初上线,即速应用在H5领域的累计,便承接在小程序上.8月7日,即速应用的用户微信 ...
- 敏捷开发冲刺--day3
1 团队介绍 团队组成: PM:齐爽爽(258) 小组成员:马帅(248),何健(267),蔡凯峰(285) Git链接:https://github.com/WHUSE2017/C-team 2 ...
- 201621123062《Java程序设计》第一周学习总结
1.本周学习总结 关键词: 初步熟悉Java的基本组成.语言特点(简单性.结构中立性).运行环境.简单语法等. 关键概念之间的联系: 1.JVM是Java程序唯一认识的操作系统,其可执行文件为.cla ...
- 小黄衫 Get
小黄衫 Get . 十分荣幸在前四次作业中以微弱的3分之差拿到了第一,获得了本次的小黄衫. 先发点牢骚.. 讲道理,原本以为的研究生生涯应该就是埋在论文堆里度过的时候顺便上上课.当初选课的时候,学 ...
- 201621123057 《Java程序设计》第12周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 2. 面向系统综合设计-图书馆管理系统或购物车 使用流与文件改造你的图书馆管理系统或购物车. 2.1 简述如何 ...
- oracle删除某个用户所有表(转)
1. select 'Drop table '||table_name||';' from all_tables where owner ...
- Python打包分发工具setuptools
作为Python标准的打包及分发工具,setuptools可以说相当地简单易用.它会随着Python一起安装在你的机器上.你只需写一个简短的setup.py安装文件,就可以将你的Python应用打包 ...
- const volatile同时限定一个类型int a = 10
const和volatile放在一起的意义在于: (1)本程序段中不能对a作修改,任何修改都是非法的,或者至少是粗心,编译器应该报错,防止这种粗心: (2)另一个程序段则完全有可能修改,因此编译器最好 ...