• 【题解】Atcoder ARC#83 E-Bichrome Tree


      哈哈~自己做出来的E题!(虽然这题被机房大佬强D极水)。最开始神经错乱,写了个完全不对的贪心,竟然只错了4个点(。•ˇ‸ˇ•。)

      可以发现,一个节点的子树内部和他颜色相同的节点权值和 是固定的,那么不一定的就是另外的那个颜色的权值和了。而由于这个图上权值可以从 0 开始取值,显然越小越好。这样就可以dp啦。

      其实我这个里面多了一个维度,就是记录当节点为黑色/白色时怎么怎么样,但其实这两个状态完全对称,根本没必要再多开一个维。但我就懒得改啦~

    #include <bits/stdc++.h>
    using namespace std;
    #define maxn 6000
    #define INF 999999999
    int n, W[maxn], f[2][2][maxn], g[maxn][2];
    
    int read()
    {
        int x = 0, k = 1;
        char c; c = getchar();
        while(c < '0' || c > '9') { if(c == '-') k = -1; c = getchar(); }
        while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
        return x * k; 
    }
    
    struct edge
    {
        int cnp, to[maxn], last[maxn], head[maxn];
        edge() { cnp = 2; }
        void add(int u, int v) { to[cnp] = v, last[cnp] = head[u], head[u] = cnp ++; }
    }E1; 
    
    bool Down(int &x, int y)
    {
        if(y >= INF) return 0;
        x = min(x, y); return 1;
    }
    
    void dfs(int u)
    {
        for(int i = E1.head[u]; i; i = E1.last[i]) { int v = E1.to[i]; dfs(v); }
        for(int i = 0; i <= 1; i ++)
            for(int j = 0; j <= W[u]; j ++) f[0][i][j] = INF;
        f[0][0][0] = f[0][1][0] = 0; int pre = 0, now = 1;
        for(int i = E1.head[u]; i; i = E1.last[i])
        {
            int v = E1.to[i];
            for(int j = 0; j <= 1; j ++)
                for(int k = 0; k <= W[u]; k ++) f[now][j][k] = INF;
            for(int k = 0; k <= 1; k ++)
                for(int j = 0; j <= W[u]; j ++)
                {
                    int flag = 0; 
                    if(j - W[v] >= 0) flag = Down(f[now][k][j], f[pre][k][j - W[v]] + g[v][k]); 
                    if(j - g[v][k ^ 1] >= 0) flag = Down(f[now][k][j], f[pre][k][j - g[v][k ^ 1]] + W[v]);
                }
            swap(pre, now);
        }
        g[u][0] = g[u][1] = INF;
        for(int k = 0; k <= 1; k ++)
            for(int j = 0; j <= W[u]; j ++)
                g[u][k] = min(g[u][k], f[pre][k][j]); 
    }
    
    int main()
    {
        n = read();
        for(int i = 2; i <= n; i ++) { int x = read(); E1.add(x, i); }
        for(int i = 1; i <= n; i ++) W[i] = read();
        dfs(1);
        if(g[1][0] < INF || g[1][1] < INF) printf("POSSIBLE
    ");
        else printf("IMPOSSIBLE
    ");
        return 0;
    }
  • 相关阅读:
    2-10 案例4:像素读取写入
    2-8 案例3:不同图片质量保存
    2-7 案例2:图片写入
    Linux文件操作实用笔记
    Linux文件操作实用笔记
    Linux 文件系统基础
    Linux 文件系统基础
    一篇故事讲述了计算机网络里的基本概念:网关,DHCP,IP寻址,ARP欺骗,路由,DDOS等...
    一篇故事讲述了计算机网络里的基本概念:网关,DHCP,IP寻址,ARP欺骗,路由,DDOS等...
    30 个与程序猿有关的成语
  • 原文地址:https://www.cnblogs.com/twilight-sx/p/9733823.html
Copyright © 2020-2023  润新知