Atcoder Beginner Contest 324 F Beautiful Path 题解-分数规划

这篇具有很好参考价值的文章主要介绍了Atcoder Beginner Contest 324 F Beautiful Path 题解-分数规划。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

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

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

题目链接

题目大意:给出一个 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 的前提下,有如下两个快速结论:文章来源地址https://www.toymoban.com/news/detail-711568.html

  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 题解-分数规划的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • AtCoder Beginner Contest 336 C - Even Digits题解

    Time Limit: 2 sec / Memory Limit: 1024 MB Score: 300300 points Problem Statement A non-negative integer �n is called a  good integer  when it satisfies the following condition: All digits in the decimal notation of �n are even numbers (00, 22, 44, 66, and 88). For example, 00, 6868, and 20242024 are good integers. You are given an integer 

    2024年01月20日
    浏览(52)
  • AtCoder Beginner Contest 302 H. Ball Collector 题解

    为了更好的阅读体验,请单击这里 AtCoder Beginner Contest 302 H. Ball Collector 题意跳过。 可以视作将 (a_i, b_i) 之间连了一条边,然后 (a_i, b_i) 之间只能选一个等价于对于一条边只能选择其一个端点。那么对于只包含树的联通块而言,如果都选择儿子节点,那么会有一个根节点无

    2024年02月06日
    浏览(39)
  • Atcoder beginner contest 336 -- E -- Digit Sum Divisible --- 题解(数位dp)

    目录   E -- Digit Sum Divisibl 题目大意: 思路解析: 代码实现: 给你一个整数n,让你找出小于等于n的数中一共有多少个好整数,并输出好整数的个数。对好整数的个数定义为如果一个数能被他的数位之和整除,则称这个数为好整数。例如 12 能被 3 整除。 n=10^14。 看到数位之和

    2024年01月16日
    浏览(62)
  • AtCoder Beginner Contest 302 H. Ball Collector 题解 可撤销并查集

    为了更好的阅读体验,请单击这里 AtCoder Beginner Contest 302 H. Ball Collector 题意跳过。 可以视作将 (a_i, b_i) 之间连了一条边,然后 (a_i, b_i) 之间只能选一个等价于对于一条边只能选择其一个端点。那么对于只包含树的联通块而言,如果都选择儿子节点,那么会有一个根节点无

    2024年02月06日
    浏览(35)
  • Atcoder Beginner Contest 321 G - Electric Circuit 题解 - 状压dp | 指定最低位

    为了更好的阅读体验,请点击这里 题目链接:G - Electric Circuit 看到了 (N) 的数据范围,因此是显然的状压 dp。 不妨设 (f_S) 为仅使用 (S) 集合中的所有点,能够连成恰好 (1) 个连通块的方案数。 (g_S) 为仅使用 (S) 集合中的所有点的方案数,其中 (cntr(S)) 在 (S) 中为

    2024年02月05日
    浏览(54)
  • AtCoder Beginner Contest 310

    感觉F又双叒叕写复杂了 点杯咖啡,要 (p) 元,但可以用一个优惠券,使得咖啡只要 (q) 元,但你需要额外购买 (n) 个商品中(价格为 (a_i) )的一个。 问点杯咖啡的最小价格。 考虑直接买还是使用优惠券,使用优惠券的话就选 (n) 个商品中价格最小的。 两种情况取最小

    2024年02月16日
    浏览(28)
  • AtCoder Beginner Contest 320

    给定 (a,b) ,输出 (a^b + b^a) 。 因为数不超过 (10) ,可以直接用 pow 计算然后转成 (int) 。不会有精度损失。 神奇的代码 给定一个字符串 (s) ,求长度最长的回文串。 因为长度只有 (100) ,可以直接枚举子串,然后暴力判断是不是回文串(翻转后是否和自身相同),复杂

    2024年02月08日
    浏览(38)
  • AtCoder Beginner Contest 348

    给定 (n) ,输出 (ooxooxoox...) ,长度为 (n) 。 按题意模拟即可。 神奇的代码 给定 (n) 个点,对每个点,求出与其距离最远的点的下标。 (n) 只有 (100) ,对于每个点,花 (O(n)) 遍历每个点最大化距离,时间复杂度为 (O(n^2)) 。 神奇的代码 (n) 个豌豆,每个豌豆有美味值

    2024年04月08日
    浏览(42)
  • AtCoder Beginner Contest 300

    给定一个元素互不相同的数组 (c) 和 (a,b) ,找到 (i) 使得 (c_i = a + b) 直接 for 循环寻找即可。 神奇的代码 给定两个矩阵 (A,B) ,问能否对 (A) 进行若干次变换操作得到 (B) 。 变换分两种,一种是将第一列放到最后一列,另一种是将第一行放到最后一行。 范围不大,直

    2024年02月01日
    浏览(58)
  • AtCoder Beginner Contest 302

    给定怪物的血量 (a) 和你每次攻击扣除的血量 (b) ,问打多少次怪物才会死。 答案即为 (lceil frac{a}{b} rceil = lfloor frac{a + b - 1}{b} rfloor) 神奇的代码 给定一个二维网格,格子上有字母,找到一连串位置,使得其字符串为 (snuke) ,要求这一连串位置俩俩相邻,即有共边或

    2024年02月05日
    浏览(78)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包