• Codeforces 1240C. Paint the Tree


    传送门

    首先每个点 $u$ 只能选择不超过 $k$ 个相连的边

    并且设边为 $(u,v)$ ,那么此时 $v$ 也必须选择这条边

    因为图是一颗树,显然考虑一下树形 $dp$

    设 $f[x][0/1]$ 表示考虑完 $x$ 的子树,当前节点有没有留一个选择给和父亲相连的边($0$ 表示没有)

    那么对于 $f[x][0]$

    考虑所有 $x$ 的儿子 $v$,我们要选出不超过 $k$ 个儿子的 $f[v][1]+val(x,v)$ ,然后剩下的儿子全部选 $f[v][0]$ ,求最大价值

    (其中 $val(x,v)$ 是边 $(x,v)$ 的价值)

    考虑一开始所有的儿子都先选 $f[v][0]$,对于某个儿子 $v$ 如果我们之后要选 $f[v][1]$ ,那么增加的贡献 $delta$ 为 $-f[v][0]+f[v][1]+val(x,v)$

    显然 $-f[v][0]$ 是因为之前已经加入了 $f[v][0]$ 的贡献

    那么此时每个儿子的选择互不影响,直接按 $delta$ 排序取前 $k$ 大即可(注意如果还没到 $k$ 个但是 $delta$ 已经小于 $0$ 了就不用选)

    对于 $f[x][1]$ 也是一样的道理,但是我们这时候取的就是前 $k-1$ 的的 $delta$ 了

    不妨设 $1$ 为根,那么答案即为 $f[1][0]$

    #include<iostream>
    #include<cstdio>
    #include<algorithm>
    #include<cstring>
    #include<cmath>
    #include<vector>
    using namespace std;
    typedef long long ll;
    inline int read()
    {
        int x=0,f=1; char ch=getchar();
        while(ch<'0'||ch>'9') { if(ch=='-') f=-1; ch=getchar(); }
        while(ch>='0'&&ch<='9') { x=(x<<1)+(x<<3)+(ch^48); ch=getchar(); }
        return x*f;
    }
    const int N=5e5+7;
    int Q,n,K;
    int fir[N],from[N<<1],to[N<<1],val[N<<1],cntt;
    inline void add(int a,int b,int c) { from[++cntt]=fir[a]; fir[a]=cntt; to[cntt]=b; val[cntt]=c; }
    ll f[N][2];
    void dfs(int x,int fa)
    {
        vector <ll> tmp;
        for(int i=fir[x];i;i=from[i])
        {
            int v=to[i]; if(v==fa) continue;
            dfs(v,x);
            f[x][0]+=f[v][0]; f[x][1]+=f[v][0];
            tmp.push_back(f[v][1]-f[v][0]+val[i]);
        }
        sort(tmp.begin(),tmp.end()); reverse(tmp.begin(),tmp.end());
        int sz=tmp.size();
        for(int i=0;i<K&&i<sz;i++)
        {
            if(tmp[i]<=0) break;
            f[x][0]+=tmp[i];
            if(i<K-1) f[x][1]+=tmp[i];
        }
    }
    int main()
    {
        Q=read();
        while(Q--)
        {
            n=read(),K=read(); int a,b,c;
            for(int i=1;i<=n;i++) f[i][0]=f[i][1]=0;
            for(int i=1;i<=n;i++) fir[i]=0; cntt=0;
            for(int i=1;i<n;i++)
            {
                a=read(),b=read(),c=read();
                add(a,b,c); add(b,a,c);
            }
            dfs(1,0);
            printf("%lld
    ",f[1][0]);
        }
        return 0;
    }
  • 相关阅读:
    MySQL 对于千万级的大表要怎么优化?
    mysql数据库优化总结
    php 正则表达式怎么匹配标签里面的style?
    php一行代码获取本周一,本周日,上周一,上周日,本月一日,本月最后一日,上月一日,上月最后一日日期
    PHP过滤常用标签的正则表达式
    php冒泡排序
    PHP中__FUNCTION__与__METHOD__的区别
    关于PHP程序员技术职业生涯规划
    初识DSP
    MATLAB图像处理基础
  • 原文地址:https://www.cnblogs.com/LLTYYC/p/11632480.html
Copyright © 2020-2023  润新知