为了更好的阅读体验,请点击这里

分数规划小技巧:尽可能将式子写成存在某种取值,使得不等式成立的形式。 不然可能需要绕几个弯才能想出来。

题目链接

题目大意:给出一个 DAG,每条边有一个 \(b_i, c_i\),保证从编号小的边向编号大的边连边,且 \(1\) 到 \(n\) 必有路径,求 \(1\) 到 \(n\) 路径上的 \(\max \frac{\sum b}{\sum c}\)。

分数规划常规做法:二分答案 \(x\),下面比较一下两种设法:

  1. \(x > \max \frac{\sum b}{\sum c} \iff\) 从 \(1\) 到 \(n\) 的所有路径都满足 \(x > \frac{\sum b}{\sum c}\) 这一条件 \(\iff\) 从 \(1\) 到 \(n\) 的所有路径都满足 \(\sum (xc - b) > 0\) 这一条件(这里必须是大于号接下来才是最短路,不然是最长路)\(\iff\) 从 \(1\) 到 \(n\) 点的最短路 \(d_n > 0\)。

    • 满足这个条件证明 \(x\) 过大,使 \(R \leftarrow M\)
    • 否则 \(L \leftarrow M\)
  2. \(x < \max \frac{\sum b}{\sum c} \iff\) 从 \(1\) 到 \(n\) 的存在某条路径满足 \(x < \frac{\sum b}{\sum c}\) 这一条件 \(\iff\) 从 \(1\) 到 \(n\) 的存在某条路径满足 \(\sum (xc - b) < 0\) 这一条件(这里如果是小于号就是天然的求最短路,如果是大于号你还需要取负修改成小于号)\(\iff\) 从 \(1\) 到 \(n\) 点的最短路 \(d_n < 0\)。
    • 满足这个条件证明 \(x\) 过小,使 \(L \leftarrow M\)
    • 否则 \(R \leftarrow M\)

设法 2 可以让你的脑子少转个圈,这样转化为存在某种方案满足这个条件,其实也是方便地用最大/最小值来验证不等式是否成立。

赛时用的设法 1,脑袋被转得晕乎乎的。

再推一点东西吧。在满足设法 2 的前提下,有如下两个快速结论:

  1. 最大化 \(\frac{\sum b}{\sum c} \Longrightarrow\) 设 \(x > \max \frac{\sum b}{\sum c} \iff \max \sum (b-cx) > 0\)

    • 存在最大值 \(\sum (b-cx) > 0 \Longrightarrow L \gets M\)
    • 否则 \(R \gets M\)
  2. 最小化 \(\frac{\sum b}{\sum c} \Longrightarrow\) 设 \(x < \min \frac{\sum b}{\sum c} \iff \min \sum (b-cx) < 0\)
    • 存在最小值 \(\sum (b-cx) < 0 \Longrightarrow R \gets M\)
    • 否则 \(L \gets M\)
#include<bits/stdc++.h>
using namespace std; typedef long long ll;
typedef long double db; #define IL inline
#define fi first
#define se second
#define mk make_pair
#define pb push_back
#define SZ(x) (int)(x).size()
#define ALL(x) (x).begin(), (x).end()
#define dbg1(x) cout << #x << " = " << x << ", "
#define dbg2(x) cout << #x << " = " << x << endl template <typename T>
void _debug(const char* format, T t) {
cerr << format << '=' << t << endl;
} template <class First, class... Rest>
void _debug(const char* format, First first, Rest... rest) {
while (*format != ',') cerr << *format++;
cerr << '=' << first << ',';
_debug(format + 1, rest...);
} template <typename T>
ostream& operator<<(ostream& os, const vector<T>& V) {
os << "[ ";
for (const auto& vv : V) os << vv << ", ";
os << ']';
return os;
}
#ifdef LOCAL
#define dbg(...) _debug(#__VA_ARGS__, __VA_ARGS__)
#else
#define dbg(...)
#endif template<typename Tp> IL void read(Tp &x) {
x=0; int f=1; char ch=getchar();
while(!isdigit(ch)) {if(ch == '-') f=-1; ch=getchar();}
while(isdigit(ch)) { x=x*10+ch-'0'; ch=getchar();}
x *= f;
}
template<typename First, typename... Rest> IL void read(First &first, Rest&... rest) {
read(first); read(rest...);
}
int buf[42];
template<typename Tp> IL void write(Tp x) {
int p = 0;
if(x < 0) { putchar('-'); x=-x;}
if(x == 0) { putchar('0'); return;}
while(x) {
buf[++p] = x % 10;
x /= 10;
}
for(int i=p;i;i--) putchar('0' + buf[i]);
}
template<typename First, typename... Rest> IL void write(const First& first, const Rest&... rest) {
write(first); putchar(32); write(rest...);
} const int N = 200000 + 5; struct E {
int u, v, b, c;
db d;
E(int u = 0, int v = 0, int b = 0, int c = 0, db d = 0.0):u(u), v(v), b(b), c(c), d(d) {}
}; const db inf = 1e18; int n, m;
db d[N];
vector<E> G[N]; bool check(db x) {
for (int i = 0; i < n; i++) {
for (int j = 0; j < G[i].size(); j++) {
E e = G[i][j];
G[i][j].d = e.c * x - e.b;
}
}
fill(d, d + n, inf);
d[0] = 0;
for (int i = 0; i < n; i++) {
for (int j = 0; j < SZ(G[i]); j++) {
E& e = G[i][j];
d[e.v] = min(d[e.v], d[i] + e.d);
}
}
return d[n-1] > 0;
} void solve() {
read(n, m);
for (int i = 0; i < m; i++) {
int u, v, b, c; read(u, v, b, c); u--; v--;
G[u].pb(E(u, v, b, c, 0.0));
}
db L = 0.0, R = 10000.0;
while (R - L > 1e-14) {
db M = (L + R) / 2.0;
if (check(M)) {R = M;}
else L = M;
}
printf("%.16Lf\n", L);
} int main() {
#ifdef LOCAL
freopen("test.in", "r", stdin);
// freopen("test.out", "w", stdout);
#endif
int T = 1;
// read(T);
while(T--) solve();
return 0;
}

Atcoder Beginner Contest 324 F Beautiful Path 题解-分数规划的更多相关文章

  1. AtCoder Beginner Contest 137 F

    AtCoder Beginner Contest 137 F 数论鬼题(虽然不算特别数论) 希望你在浏览这篇题解前已经知道了费马小定理 利用用费马小定理构造函数\(g(x)=(x-i)^{P-1}\) ...

  2. AtCoder Beginner Contest 215 F题题解

    F - Dist Max 2 什么时候我才能突破\(F\)题的大关... 算了,不说了,看题. 简化题意:给定\(n\)个点的坐标,定义没两个点的距离为\(min(|x_i-x_j|,|y_i-y_j ...

  3. AtCoder Beginner Contest 213 F题 题解

    F - Common Prefixes 该题也是囤了好久的题目了,看题目公共前缀,再扫一眼题目,嗯求每个后缀与其他后缀的公共前缀的和,那不就是后缀数组吗?对于这类问题后缀数组可是相当在行的. 我们用后 ...

  4. AtCoder Beginner Contest 261 F // 树状数组

    题目链接:F - Sorting Color Balls (atcoder.jp) 题意: 有n个球,球有颜色和数字.对相邻的两球进行交换时,若颜色不同,需要花费1的代价.求将球排成数字不降的顺序,所 ...

  5. AtCoder Beginner Contest 260 F - Find 4-cycle

    题目传送门:F - Find 4-cycle (atcoder.jp) 题意: 给定一个无向图,其包含了S.T两个独立点集(即S.T内部间的任意两点之间不存在边),再给出图中的M条边(S中的点与T中的 ...

  6. AtCoder Beginner Contest 253 F - Operations on a Matrix // 树状数组

    题目传送门:F - Operations on a Matrix (atcoder.jp) 题意: 给一个N*M大小的零矩阵,以及Q次操作.操作1(l,r,x):对于 [l,r] 区间内的每列都加上x ...

  7. AtCoder Beginner Contest 249 F - Ignore Operations // 贪心 + 大根堆

    传送门:F - Keep Connect (atcoder.jp) 题意: 给定长度为N的操作(ti,yi). 给定初值为0的x,对其进行操作:当t为1时,将x替换为y:当t为2时,将x加上y. 最多 ...

  8. AtCoder Beginner Contest 247 F - Cards // dp + 并查集

    原题链接:F - Cards (atcoder.jp) 题意: 给定N张牌,每张牌正反面各有一个数,所有牌的正面.反面分别构成大小为N的排列P,Q. 求有多少种摆放方式,使得N张牌朝上的数字构成一个1 ...

  9. AtCoder Beginner Contest 143 F - Distinct Numbers

    题意 给出一个长度为NNN的序列,求对于所有k∈[1,N]k\in[1,N]k∈[1,N],每次从序列中选出kkk个互不相同的数,最多能取多少次. N≤3e5N\le3e5N≤3e5 题解 我们首先把 ...

  10. AtCoder Beginner Contest 133 F Colorful Tree

    Colorful Tree 思路: 如果强制在线的化可以用树链剖分. 但这道题不强制在线,那么就可以将询问进行差分,最后dfs时再计算每个答案的修改值, 只要维护两个数组就可以了,分别表示根节点到当前 ...

随机推荐

  1. Uncaught (in promise) NavigationDuplicated: Avoided redundant navigation to current location: "/xxx". at createRouterError 的说明和解决

    错误说明 Uncaught (in promise) NavigationDuplicated: Avoided redundant navigation to current location: & ...

  2. 和 ChatGPT 聊聊 .NET 编译和执行背后的那些事儿

    1 .NET 编译.构建.执行涉及到哪些概念 在 .NET 编译.构建和执行中,涉及到以下概念: C# 或 Visual Basic .NET 等编程语言: 这些是 .NET Framework 使用 ...

  3. Git基本操作命令大全

    一.全局配置命令 ## 配置级别: –local(默认,高级优先):只影响本地仓库 –global(中优先级):只影响所有当前用户的git仓库 –system(低优先级):影响到全系统的git仓库 # ...

  4. WordPress对url做重定向处理

    在一个网站进行改版滞后,可能会产生大量的错误的urls,我们想让这些urls还是存在的,并跳转到新的url,此时就要做301重定向. 针对wordpress做重定向,一般推荐使用redirection ...

  5. 计算机组成原理—中央处理器CPU

    文章目录 CPU的功能与架构 CPU的组成 运算器 控制器 指令执行过程 指令流程 指令执行方案 数据通路 单总线结构 专用通路结构 硬布线控制器设计 硬布线执行流程 硬布线CU内部 怎么设计微操作的 ...

  6. C语言:删除顺序表中重复的信息—(删除顺序表中重复的单词)

    如何删除顺序表中的重复单词: (开始看内容之前容朕说一句:如果你最后怎么都运行不了你想要的结果,①我敢保证大概率是你的下标越界你的下标越界了你的下标越界了.②在我这程序里面你肯定打少了p--,少了p- ...

  7. sass 导入@import详解

    @import ​ Sass 拓展了 @import 的功能,允许其导入 SCSS 或 Sass 文件.被导入的文件将合并编译到同一个 CSS 文件中,另外,被导入的文件中所包含的变量或者混合指令 ( ...

  8. linux file命令查看文件类型

    在linux系统中,linux是不根据后缀名识别文件类型的,所以使用file命令查看文件的类型. [root@node5 ~]# file /etc/shadow /etc/shadow: ASCII ...

  9. 高分辨率食道测压(HRM)

    高分辨率测压(High resolution Manometry) HRM的优势 高分辨率食管测压不但实现了从咽部到胃部的全程功能监测,而且插管无需牵拉,操作十分方便.更为重要的是,临床医生经过简单的 ...

  10. linux 自定义程序开机自启

    实现开机自启常见的有两种方法: /etc/init.d/下编写脚本命令(有些机子会有问题,比较麻烦) 利用定时任务crontab 本文介绍crontab现实程序开机自启 编写执行脚本run.sh #! ...