首页 > 其他 > 详细

BZOJ[3252]攻略(长链剖分)

时间:2018-10-07 21:08:13      阅读:148      评论:0      收藏:0      [点我收藏+]

BZOJ[3252]攻略

Description

题目简述:树版[k取方格数]

众所周知,桂木桂马是攻略之神,开启攻略之神模式后,他可以同时攻略k部游戏。今天他得到了一款新游戏《XX半岛》,这款游戏有n个场景(scene),某些场景可以通过不同的选择支到达其他场景。所有场景和选择支构成树状结构:开始游戏时在根节点(共通线),叶子节点为结局。每个场景有一个价值,现在桂马开启攻略之神模式,同时攻略k次该游戏,问他观赏到的场景的价值和最大是多少(同一场景观看多次是不能重复得到价值的)

“为什么你还没玩就知道每个场景的价值呢?”

“我已经看到结局了。”

Input

第一行两个正整数n,k

第二行n个正整数,表示每个场景的价值

以下n-1行,每行2个整数a,b,表示a场景有个选择支通向b场景(即a是b的父亲)

保证场景1为根节点

n<=200000,1<=场景价值<=2^31-1

Output

输出一个整数表示答案

Sample Input

5 2
4 3 2 1 1
1 2
1 5
2 3
2 4

Sample Output

10

一个类似长链剖分的东西。
直接把链按照权值剖分,取前k个最长的链

#include<bits/stdc++.h>
#define lll long long
using namespace std;
lll read(){
    lll x=0,w=1;char ch=getchar();
    while(ch>'9'||ch<'0'){if(ch=='-')w=-1;ch=getchar();}
    while(ch>='0'&&ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar();
    return x*w;
}
const lll N=200010;
lll n,k,cnt,opt,ans,x,y;
lll a[N],head[N],son[N],sum[N],c[N];
struct node{
    lll to,next;
}edge[N];
bool cmp(lll p,lll q){return p>q;}
void add(lll x,lll y){
    cnt++;edge[cnt].to=y;edge[cnt].next=head[x];head[x]=cnt;
}
void dfs1(lll k){
    for(lll i=head[k];i;i=edge[i].next){
        lll v=edge[i].to;dfs1(v);
        sum[k]=max(sum[k],sum[v]);if(sum[v]>sum[son[k]])son[k]=v;
    }sum[k]+=a[k];
}
void dfs2(lll k,lll top){
    if(k==top)c[++opt]=sum[k];
    if(!son[k])return;
    dfs2(son[k],top);
    for(lll i=head[k];i;i=edge[i].next){
        lll v=edge[i].to;if(v==son[k])continue;
        dfs2(v,v);
    }
}
int main(){
    n=read();k=read();
    for(lll i=1;i<=n;i++)a[i]=read();
    for(lll i=1;i<n;i++){
        x=read();y=read();add(x,y);
    }dfs1(1);dfs2(1,1);sort(c+1,c+1+opt,cmp);
    for(lll i=1;i<=k;i++)ans+=c[i];
    printf("%lld\n",ans);return 0;
}

BZOJ[3252]攻略(长链剖分)

原文:https://www.cnblogs.com/lsgjcya/p/9751394.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!