zoukankan      html  css  js  c++  java
  • [国家集训队]阿狸和桃子的游戏

    对于两点之间的边是不好处理的,但基于观察可以发现:双方都只会将点染色,因此我们的想法是将边权放到点权上去然后构造出一种选择点权的方案与原问题等价。

    同时,因为对于一条边而言它连接的两点代价是一致的,因此下放点权的过程应该满足对称性。

    此时基于观察可以发现:

    • (forall (u, v, w)) 如果下放到 (u, v) 的点权满足对称性即相等,则如果 (u, v) 没有同时被一个人选择那么边权将会恰好抵消满足题意。

    那么我们只需要调整下方的大小使得当 (u, v) 同时被选时获得的权值恰好边权即可。

    不难发现有且仅有 (frac{w}{2}) 满足要求。

    但因为 (w) 可能为奇数不方便计算,因此我们开始将所有权值扩大一倍最后将答案减半即可。

    于是问题就被转化为:先手后手每次可以取一个点获得这个点的点权,问对抗博弈后先手减后手的得分。

    不难发现此时先后手的策略都一定是取每次剩下的最大值,排序一遍即可,复杂度 (O(n log n + m))

    #include <bits/stdc++.h>
    using namespace std;
    #define rep(i, l, r) for (int i = l; i <= r; ++i)
    const int N = 10000 + 5;
    int n, m, u, v, w, ans, a[N];
    int read() {
        char c; int x = 0, f = 1;
        c = getchar();
        while (c > '9' || c < '0') { if(c == '-') f = -1; c = getchar();}
        while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
        return x * f;
    }
    signed main() {
        n = read(), m = read();
        rep(i, 1, n) a[i] = read() * 2;
        rep(i, 1, m) u = read(), v = read(), w = read(), a[u] += w, a[v] += w;
        sort(a + 1, a + n + 1);
        for (int i = n, j = 1; i >= 1; --i, j = -j) ans += j * a[i];
        printf("%d", ans / 2);
        return 0;
    }
    
  • 相关阅读:
    .net编程扫盲(*)
    接口编程扫盲(多态)
    (转)栈与堆栈的区别
    (转).NET基础拾遗(5)多线程开发基础
    (转)你应该知道的计算机网络知识
    网络代理的基础知识
    某代理网站免费IP地址抓取测试
    常用Maven插件介绍
    Maven打jar发布包的常用配置
    Apache Commons CLI 开发命令行工具示例
  • 原文地址:https://www.cnblogs.com/Go7338395/p/13865637.html
Copyright © 2011-2022 走看看