//代码跑的很慢四秒会超时,结尾附两秒代码(标程)
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const ll inf=0x3f3f3f3f;
vector<pair<int,int>>tree[100010];
ll dp[100010][110];/*dp[u][i]表示u节点为根的子树上选择了i个叶子节点,
会经过u这个子树的边的权值和的最优值。转移方程如下:
dp[u][i]=min( dp[u][i-j]+dp[v][j]+w*j*(k-j) );
v是u的子节点,w是u与v之间边的权值,l*(k-l)表示的是这条边会被经过的次数
(j为从v这个树上过来的叶节点数,k-j为其它地方选择的叶节点数)*/
int siz[100010];
int t,n,k;
void dfs(int u,int f)
{
    if(tree[u].size()==1)
    {
        dp[u][0]=0;
        dp[u][1]=0;
        siz[u]=1;//记录叶子节点数量
        return;
    }
    siz[u]=0;
    dp[u][0]=0;
    for(int i=0;i<tree[u].size();i++)
    {
        int v=tree[u][i].first;//联接点
        int w=tree[u][i].second;//权重
        if(v==f)
            continue;
        dfs(v,u);
        siz[u]+=siz[v];
        dp[u][k]=min(dp[u][k],dp[v][k]);//在v上选了k个叶子
        for(int j=min(k,siz[u]);j>=1;j--)//限制j最大为k
        {
            for(int l=1;l<=min(j,siz[v]);l++)
                dp[u][j]=min(dp[u][j],dp[u][j-l]+dp[v][l]+l*(k-l)*w);//状态转移方程
        }
    }
}
int main()
{
    scanf("%d",&t);
    for(int tt=1;tt<=t;tt++)
    {
        scanf("%d%d",&n,&k);
        for(int i=1;i<=n;i++)
            tree[i].clear();
        for(int i=1;i<n;i++)
        {
            int u,v;
            int w;
            scanf("%d%d%lld",&u,&v,&w);
            tree[u].push_back({v,w});
            tree[v].push_back({u,w});
        }
        printf("Case #%d: ",tt);
        if(n==2)
            printf("%d\n",(k==2?tree[1][0].second:0));//当且仅当k==2时存在一条路的长度
        else
        {
            int rt=0;
            for(int i=1;i<=n;i++)
                if(tree[i].size()>1)//找到一个不为叶子节点的节点作根
                {
                    rt=i;
                    break;
                }
            for(int i=1;i<=n;i++)
            {
                dp[i][0]=0;
                for(int j=1;j<=k;j++)
                    dp[i][j]=inf;
            }
            dfs(rt,0);
            printf("%lld\n",dp[rt][k]);
        }
    }
    return 0;
}
 /*标程AC代码
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAXN=100005;
const int MAXK=105;
const ll INF=(1LL<<60)-1;
vector<pair<int,int> > e[MAXN];
ll dp[MAXN][MAXK],tmp[MAXK];
int sz[MAXN];
void dfs(int u,int f,int k)
{
    if((int)e[u].size()==1)
    {
        sz[u]=1,dp[u][0]=dp[u][1]=0;
        return;
    }
    sz[u]=0;
    for(auto &t:e[u])
    {
        int v=t.first,c=t.second;
        if(v==f)continue;
        dfs(v,u,k);
        int ts=min(k,sz[u]+sz[v]);
        for(int i=0;i<=ts;i++)
            tmp[i]=INF;
        for(int i=0;i<=sz[u];i++)
            for(int j=0;j<=sz[v] && i+j<=ts;j++)
                tmp[i+j]=min(tmp[i+j],dp[u][i]+dp[v][j]+j*(k-j)*c);
        for(int i=0;i<=ts;i++)
            dp[u][i]=tmp[i];
        sz[u]=ts;
    }
}
int main()
{
    int T;
    scanf("%d",&T);
    for(int ca=1;ca<=T;ca++)
    {
        int n,k;
        scanf("%d%d",&n,&k);
        for(int i=1;i<=n-1;i++)
        {
            int u,w,v;
            scanf("%d%d%d",&u,&v,&w);
            e[u].push_back({v,w});
            e[v].push_back({u,w});
        }
        printf("Case #%d: ",ca);
        if(n==2)printf("%d\n",(k==2 ? e[1][0].second : 0));
        else
        {
            int rt=0;
            for(int i=1;i<=n;i++)
                if((int)e[i].size()>1)rt=i;
            assert(rt);
            dfs(rt,0,k);
            printf("%lld\n",dp[rt][k]);
        }
        for(int i=1;i<=n;i++)
            e[i].clear();
    }
    return 0;
}
*/

2018宁夏邀请赛G(DFS,动态规划【VECTOR<PAIR>】)的更多相关文章

  1. 2018宁夏邀请赛 L Continuous Intervals(单调栈+线段树)

    2018宁夏邀请赛 L Continuous Intervals(单调栈+线段树) 传送门:https://nanti.jisuanke.com/t/41296 题意: 给一个数列A 问在数列A中有多 ...

  2. “2018宁夏邀请赛 ” 兼 “The 2019 Asia Yinchuan First Round Online Programming”

    ------------7题弟弟,被各位半小时13题的大佬打惨了(滑稽)---------- 签到题就不写了. F :Moving On            (1247ms) 题意:给定大小为N的带 ...

  3. 2018宁夏邀请赛网赛 I. Reversion Count(java练习题)

    题目链接 :https://nanti.jisuanke.com/t/26217 Description: There is a positive integer X, X's reversion c ...

  4. [ICPC 2018 宁夏邀请赛] A-Maximum Element In A Stack(思维)

    >传送门< 前言 辣鸡网络赛,虽然我是个菜鸡,然而好几个队伍十几分钟就AK???我心态那会彻底崩了,后来群里炸了,话题直接上知乎热搜,都是2018ICPC宁夏网络赛原题,这怎么玩,拼手速? ...

  5. 2018宁夏邀请赛L Continuous Intervals

    题目链接:https://nanti.jisuanke.com/t/28412 题意: 给出n个数的序列.问序列中有多少个区间满足,排序完之后任意两个相邻的数之差不大于1. 题解: 用max表示区间最 ...

  6. 2018宁夏邀请赛K Vertex Covers

    题目链接:https://nanti.jisuanke.com/t/28411 题意: 给出n(n<=36)个点的一个图.求点覆盖集数. 题解: 将n个点折半为L和R两部分.对于R内部的边,枚举 ...

  7. 2018宁夏邀请赛I题 bubble sort(思维题

    https://vjudge.net/problem/Gym-102222I 居然补到个防ak,刚开始不知道啥是循环左移右移(只能移一次),不好想.. 题意:以冒泡排序为背景 给你n,k 问在1~n的 ...

  8. 2018宁夏邀请赛 Continuous Intervals(单调栈 线段树

    https://vjudge.net/problem/Gym-102222L 题意:给你n个数的序列,让判断有几个区间满足排完序后相邻两数差都不大于1. 题解:对于一个区间 [L,R],记最大值为 m ...

  9. 2018宁夏邀请赛K题Vertex Covers(高维前缀和 状压 折半

    https://vjudge.net/problem/Gym-102222K 题意:给定N点M边的无向图,每个点有点权.  点覆盖表示某个点集S{}覆盖了所有的边,其贡献是S中点权之积. 现在让你求所 ...

随机推荐

  1. BZOJ 3391 [Usaco2004 Dec]Tree Cutting网络破坏:dfs【无根树 节点分枝子树大小】

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3391 题意: 给你一棵无根树,求分支size均不大于一半点数的点. 题解: 假定1为根. ...

  2. Python中深拷贝与浅拷贝区别

    浅拷贝, list值是可变的,str值不可变,只能重新赋值 a=b=c='wjx'print(a,b,c)c= 'jmy'#重新赋值了,所以内存分配了新的地址print(a,b,c)print(id( ...

  3. android:layout_weight的真实含义/android:layout_gravity的条件

    用layout_weight的时候,不要把宽度(或是高度,你想分配weight的那个)设成match_parent. android:layout_weight只适用于LinearLayout and ...

  4. jQuery插件:图片放大镜--jQuery Zoom

    本文转载于http://blog.csdn.net/xinhaozheng/article/details/4085644, 这是一款非常不错的给图片添加放大镜效果,可以应用在诸如zen cart,m ...

  5. freeMarker(四)——模板开发指南之模板

    学习笔记,选自freeMarker中文文档,译自 Email: ddekany at users.sourceforge.net 模板开发指南之模板 1. 总体结构 实际上用程序语言编写的程序就是模板 ...

  6. SPOJ705 Distinct Substrings (后缀自动机&后缀数组)

    Given a string, we need to find the total number of its distinct substrings. Input T- number of test ...

  7. HDU5446 Unknown Treasure(组合数膜合数-->Lucas+中国剩余定理)

    >On the way to the next secret treasure hiding place, the mathematician discovered a cave unknown ...

  8. bzoj 3307: 雨天的尾巴 线段树合并

    题目大意: N个点,形成一个树状结构.有M次发放,每次选择两个点x,y对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.问完成所有发放后,每个点存放最多的是哪种物品. 题解: 首先我们为每一个节 ...

  9. LVS实战1

    (一).NAT模式:NAT模型:地址转换类型,主要是做地址转换,类似于iptables的DNAT类型,它通过多目标地址转换,来实现负载均衡:特点和要求: 1.LVS(Director)上面需要双网卡: ...

  10. js之翻牌游戏中的一个深刻感悟

    先“上菜”: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UT ...