zoukankan      html  css  js  c++  java
  • 【BZOJ2395】[Balkan 2011]Timeismoney

    【BZOJ2395】[Balkan 2011]Timeismoney

    题面

    (darkbzoj)

    题解

    如果我们只有一个条件要满足的话直接最小生成树就可以了,但是现在我们有两维啊。。。

    我们将每个方法的费用和时间看作一个二维坐标((x,y))

    则我们要求(xcenterdot y=k)最小即要求反比例函数(y=frac kx)的图像离坐标轴最近。

    那么我们怎么求呢?分下面三步:

    • (Step1)

    分别求出离(y,x)轴最近的点,这个通过直接最小生成树一维排序可以求出。

    • (Step2)

    现在我们的情况是这样的:

    img

    (A),(B)是我们刚才固定的,现在我们需要找到一个点(C),使(C)(AB)左侧且离(AB)最远。

    这个怎么办呢?

    其实就是让(S_{ riangle ABC})的面积最大。
    因为有

    [vec {AB}=(B.x-A.x,B.y-A.y)\ vec {AC}=(C.x-A.x,C.y-A.y) ]

    且有(S_{ riangle ABC}=frac {vec{AB}*vec{AC}}2)(但是这个面积是有向的且为负,所以要使这两个向量乘起来最小)

    所以我们要最小化:

    [vec{AB}*vec{AC}\ =(B.x-A.x)*(C.y-A.y)-(B.y-A.y)*(C.x-A.x)\ =(B.x-A.x)*C.y+(A.y-B.y)*C.x+... ]

    因为我懒后面部分是常数,所以省略了。

    那么我直接将改边权为((B.x-A.x)y[i]+(A.y-B.y)x[i]),做最小生成树即可。

    • (Step3)

    现在找到了(C),递归处理((A,C))((C,B))即可。

    终止条件(vec {BA} * vec{CA}geq0),说明此时(C)已经跑到(AB)右侧去了。

    代码

    #include <iostream> 
    #include <cstdio> 
    #include <cstdlib> 
    #include <cstring> 
    #include <cmath> 
    #include <algorithm> 
    using namespace std;  
    const int MAX_N = 205; 
    const int MAX_M = 1e4 + 5; 
    struct Point { int x, y; } ;
    Point ans = {(int)1e9, (int)1e9}; 
    Point operator - (const Point &l, const Point &r) { return (Point){l.x - r.x, l.y - r.y}; } 
    int cross(const Point &l, const Point &r) { return l.x * r.y - l.y * r.x; } 
    int N, M, pa[MAX_N], rnk[MAX_N]; 
    int getf(int x) { return pa[x] == x ? x : pa[x] = getf(pa[x]); } 
    inline void unite(int x, int y) { 
        x = getf(x), y = getf(y); 
        if (rnk[x] == rnk[y]) pa[x] = y, ++rnk[y]; 
        else (rnk[x] < rnk[y]) ? (pa[x] = y) : (pa[y] = x); 
    } 
    struct Edge { int u, v, c, t, w; } e[MAX_M]; 
    inline bool operator < (const Edge &l, const Edge &r) { return l.w < r.w; }
    #define RG register 
    inline Point kruskal() { 
        Point res = (Point){0, 0}; 
        int tot = 0; 
        for (RG int i = 1; i <= N; ++i) pa[i] = i, rnk[i] = 1; 
        sort(&e[1], &e[M + 1]); 
        for (RG int i = 1; i <= M; ++i) { 
            int u = getf(e[i].u), v = getf(e[i].v); 
            if (u != v) unite(u, v), res.x += e[i].c, res.y += e[i].t, ++tot;
            if (tot == N - 1) break; 
        } 
        long long Ans = 1ll * ans.x * ans.y, now = 1ll * res.x * res.y; 
        if (Ans > now || (Ans == now && res.x < ans.x)) ans = res; 
        return res; 
    } 
    void Div(const Point &A, const Point &B) { 
        for (RG int i = 1; i <= M; ++i) e[i].w = e[i].t * (B.x - A.x) + e[i].c * (A.y - B.y); 
        Point C = kruskal(); 
        if (cross(B - A, C - A) >= 0) return ; 
        Div(A, C); Div(C, B); 
    } 
    int main() { 
    #ifndef ONLINE_JUDGE 
        freopen("cpp.in", "r", stdin); 
    #endif
        scanf("%d%d", &N, &M); 
        for (RG int i = 1; i <= M; i++) scanf("%d%d%d%d", &e[i].u, &e[i].v, &e[i].c, &e[i].t); 
        for (RG int i = 1; i <= M; i++) ++e[i].u, ++e[i].v, e[i].w = e[i].c; 
        Point A = kruskal(); 
        for (RG int i = 1; i <= M; i++) e[i].w = e[i].t; 
        Point B = kruskal(); 
        Div(A, B); 
        printf("%d %d
    ", ans.x, ans.y); 
        return 0; 
    } 
    

    为什么标签有个凸包呢?因为你满足要求的决策点肯定在凸包上啊

  • 相关阅读:
    Azure IoT 技术研究系列4-Azure IoT Hub的配额及缩放级别
    Azure IoT 技术研究系列3-设备到云、云到设备通信
    Azure IoT 技术研究系列2-设备注册到Azure IoT Hub
    Azure IoT 技术研究系列1-入门篇
    消息队列技术之基本概念
    PAT甲级题分类汇编——线性
    AVR单片机教程——闪烁LED
    AVR单片机教程——点亮第一个LED
    AVR单片机教程——开发环境配置
    AVR单片机教程——开发板介绍
  • 原文地址:https://www.cnblogs.com/heyujun/p/10398181.html
Copyright © 2011-2022 走看看