首页 > 其他分享 >Atcoder Beginner Contest 324 F Beautiful Path 题解-分数规划

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

时间:2023-10-14 23:01:32浏览次数:35  
标签:Beautiful Atcoder int 题解 sum db read void define

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

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

题目链接

题目大意:给出一个 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,脑袋被转得晕乎乎的。

#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;
}

标签:Beautiful,Atcoder,int,题解,sum,db,read,void,define
From: https://www.cnblogs.com/bringlu/p/17764921.html

相关文章

  • CF1204D2 Kirk and a Binary String (hard version) 题解
    CF1204D2KirkandaBinaryString(hardversion)题解分析先来分析\(01\)串的最长不下降子序列。全是\(0\)显然是不下降的,如果中间出现一个\(1\),为了维护不下降的性质,后面就只能全是\(1\)。一句话概括一下,\(0\)后面能跟\(0,1\),\(1\)后面只能跟\(1\)。现在来分析这......
  • 题解 [ABC258G] Triangle
    题目链接\(\rmO(n^3)\)枚举\(i,j,k\)的算法是显然的。考虑优化掉一个\(n\),如果枚举\(i,j\),那么显然需要找出有多少个\(k\)同时满足\(a_{i,k}=a_{j,k}=1\),我们可以将\(a_i\)和\(a_j\)看作两个二进制数,那么同时等于\(1\)的位置就是并起来等于\(1\)的位置,\(bitset......
  • 观光奶牛 详细题解
    #T3#SPFA判断正/负环#二分查找为啥现在突然发出来:翻自个笔记发现这篇写的挺好hhh361.观光奶牛-AcWing题库给定一张\(L\)个点、\(P\)条边的有向图,每个点都有一个权值\(f[i]\),每条边都有一个权值\(t[i]\)。求图中的一个环,使“环上各点的权值之和”除以“环上各边的......
  • [题解]AT_abc153_f [ABC153F] Silver Fox vs Monster
    模拟赛最后\(15\)分钟想到的做法。思路首先有一个显然的贪心策略:我们放炸弹的地方要尽可能的使这个炸弹能影响到更多的怪上。那么我们可以将对于一个怪\(i\)能够影响到它的区间表示出来\([\max(1,l_i-d),a_i+r]\)。然后将这些区间排个序,可以粗略画出这样的图:根据上......
  • P1084疫情控制 题解
    P1084疫情控制前言:这题思路不难,实现稍微有点难。总体来说,不算特别难的那种紫题,建议评蓝。题目描述给定一些点,用这些点来切断根节点到所有叶子节点的路径,可以移动这些点,不同的点可以同时移动,求时间最少。思考过程不同的点可以同时移动:看到这里,我们可以转化一下题目:给定......
  • [AGC033C] Removing Coins题解
    思路可以看出,每次对一个点\(u\)操作一次,就相当于删除以\(u\)为根的所有叶节点。当然我们还是没有什么思路,我们可以想简单一点:在一条链上的情况。如果\(u\)是链的端点:以\(u\)为根节点的叶节点只有一个,所以链的长度减一。如果\(u\)不是链的端点:以\(u\)为根节点......
  • P1612 [yLOI2018] 树上的链 题解
    思路看到条件\(2\),我们得知:这个节点对应的最长链,一定在这个节点到根节点的简单路径上。所以我们记录\(1\)到\(i\)之间的权值和,记为\(sum_i\)。因为权值是正整数,所以满足单调性,可以二分。如何二分路径上的点呢?我们维护一个与当前dfs同步的栈,记录从根节点到当前节点的简......
  • [ARC116C] Multiple Sequences题解
    思路我们可以很好的想到一种\(O(nm)\)的dp:状态:\(dp_{i,j}\)为搜到第\(i\)个,最后一个数是\(j\)的方案数。转移:\(dp_{i,j}=\displaystyle\sum_{k|j,k\not=j}dp_{i-1,k}\)当然这是会超时的。我们换一种思路,我们先枚举最后一个数,再计算方案数。这有个好处,我们缩小......
  • 苏格拉底问答、实践过程截图、遇到问题解决问题截图,代码链接
    defineVOLUME_NAME"EXT2FS"//卷名#defineBLOCK_SIZE512//块大小#defineDISK_SIZE4612//磁盘总块数defineDISK_START0//磁盘开始地址#defineSB_SIZE32//超级块大小是32BdefineGD_SIZE32//块组描述符大小是32B#defineGDT_START(0+512)//块组描述......
  • AtCoder Beginner Contest 321 C-321-like Searcher
    可以观察到0-9的所有子集都能恰组成一个满足题目条件的数字,所以共有1022个数{除空集和0}方法就是二元枚举,找出所有数然后排序。#include<iostream>#include<cstdio>#include<vector>#include<algorithm>usingnamespacestd;usingll=longlong;vector<ll>v;sig......