题解

显然权值都是正的,我们最深的那个点一定延伸到了某个叶子

我们抛去这条链之外再选K个点即可

如果直接对一棵树选K个点,满足这样的依赖关系,可以通过一个后序遍历的顺序做出来

转移方法是

\(dp[i][j] = dp[i - 1][k] + (j - k) * v\)

或者

\(dp[i][j] = dp[i - siz[u]][j]\)

代表这个点选或者不选

我们把每个点拆成1和a[i] - 1两个点,然后做两次儿子遍历顺序恰好相反的dp

我们枚举一个叶子的时候,在这个点右侧这两个后序遍历重合的地方只有这个叶子到根所有点,这也是我们要必选的点

然后我们用两个遍历中这个点左侧的点集,枚举每个点集选几个,来更新答案即可

代码

#include <bits/stdc++.h>
#define enter putchar('\n')
#define space putchar(' ')
#define fi first
#define se second
#define MAXN 40005
//#define ivorysi
#define pii pair<int,int>
#define pb push_back
using namespace std;
typedef long long int64;
template<class T>
void read(T &res) {
res = 0;char c = getchar();T f = 1;
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 + c - '0';
c = getchar();
}
res *= f;
} template<class T>
void out(T x) {
if(x < 0) {putchar('-');x = -x;}
if(x >= 10) out(x / 10);
putchar('0' + x % 10);
}
struct node {
int to,next;
}E[MAXN * 2];
int64 tot;
int N,K,head[MAXN],sumE,fa[MAXN],Ncnt,v[MAXN],a[MAXN],sumv[MAXN];
int siz[MAXN],dep[MAXN],LA[MAXN],posA[MAXN],LB[MAXN],posB[MAXN],idx;
int f[51000005],g[51000005],Q1[500005],Q2[500005],ql,qr;
bool lef[MAXN];
void add(int u,int v) {
E[++sumE].to = v;
E[sumE].next = head[u];
head[u] = sumE;
}
void dfsa(int u) {
siz[u] = 1;dep[u] = dep[fa[u]] + 1;sumv[u] = sumv[fa[u]] + v[u];
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;
dfsa(v);siz[u] += siz[v];
}
LA[++idx] = u;posA[u] = idx;
}
void dfsb(int u) {
vector<int> son;son.clear();
for(int i = head[u] ; i ; i = E[i].next) {
int v = E[i].to;son.pb(v);
}
reverse(son.begin(),son.end());
for(int i = 0 ; i < son.size() ; ++i) dfsb(son[i]);
LB[++idx] = u;posB[u] = idx;
}
void DP(int *L,int *p,int *f) {
for(int i = 1 ; i <= Ncnt ; ++i) {
int u = L[i];
int *f1 = f + (i - 1) * (K + 1),*f2 = f + i * (K + 1);
memcpy(f2,f + (p[u] - siz[u]) * (K + 1),sizeof(int) * (K + 1));
if(u <= N) {
for(int j = 1 ; j <= K ; ++j) f2[j] = max(f2[j],f1[j - 1] + v[u]);
}
else {
Q1[ql = qr = 1] = 0;Q2[1] = 0;
for(int j = 1 ; j <= K ; ++j) {
while(ql <= qr && j - Q1[ql] > a[u]) ++ql;
if(ql <= qr) f2[j] = max(f2[j],Q2[ql] + j * v[u]);
while(ql <= qr && f1[j] - j * v[u] >= Q2[qr]) --qr;
Q2[++qr] = f1[j] - j * v[u];Q1[qr] = j;
}
}
}
}
void Init() {
memset(head,0,sizeof(head));
memset(lef,0,sizeof(lef));
sumE = 0;tot = 0;
memset(f,0,sizeof(int) * Ncnt * (K + 1));
memset(g,0,sizeof(int) * Ncnt * (K + 1));
read(N);read(K);
Ncnt = N;
for(int i = 1 ; i <= N ; ++i) {
read(fa[i]);read(a[i]);read(v[i]);
if(fa[i] != 0) add(fa[i],i);
lef[fa[i]] = 1;
tot += a[i];
if(a[i] > 1) {
add(i,++Ncnt);fa[Ncnt] = i;v[Ncnt] = v[i];a[Ncnt] = a[i] - 1;a[i] = 1;
}
}
idx = 0;dfsa(1);
idx = 0;dfsb(1);
}
void Solve() {
DP(LA,posA,f);
DP(LB,posB,g);
int ans = 0;
for(int u = 1 ; u <= N ; ++u) {
if(!lef[u]) {
int *df = f + (K + 1) * (posA[u] - 1),*dg = g + (K + 1) * (posB[u] - siz[u]);
int64 cur = min((int64)K,tot - dep[u]);
for(int j = 0 ; j <= cur ; ++j) {
ans = max(ans,df[j] + dg[cur - j] + sumv[u]);
}
}
}
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
int T;
read(T);
while(T--) {
Init();
Solve();
}
return 0;
}

感觉一看到SDOI R2自动降智

这么难的省选题要是我去考根本考不动啊

【LOJ】#2268. 「SDOI2017」苹果树的更多相关文章

  1. loj#2269. 「SDOI2017」切树游戏

    还是loj的机子快啊... 普通的DP不难想到,设F[i][zt]为带上根玩出zt的方案数,G[i][zt]为子树中的方案数,后面是可以用FWT优化的 主要是复习了下动态DP #include< ...

  2. loj#2002. 「SDOI2017」序列计数(dp 矩阵乘法)

    题意 题目链接 Sol 质数的限制并没有什么卵用,直接容斥一下:答案 = 忽略质数总的方案 - 没有质数的方案 那么直接dp,设\(f[i][j]\)表示到第i个位置,当前和为j的方案数 \(f[i ...

  3. @loj - 2004@ 「SDOI2017」硬币游戏

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 周末同学们非常无聊,有人提议,咱们扔硬币玩吧,谁扔的硬币正面次数 ...

  4. LOJ #2005. 「SDOI2017」相关分析 线段树维护回归直线方程

    题目描述 \(Frank\) 对天文学非常感兴趣,他经常用望远镜看星星,同时记录下它们的信息,比如亮度.颜色等等,进而估算出星星的距离,半径等等. \(Frank\) 不仅喜欢观测,还喜欢分析观测到的 ...

  5. LibreOJ 2003. 「SDOI2017」新生舞会 基础01分数规划 最大权匹配

    #2003. 「SDOI2017」新生舞会 内存限制:256 MiB时间限制:1500 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: 匿名 提交提交记录统计讨论测试数据   题目描述 ...

  6. Loj #2192. 「SHOI2014」概率充电器

    Loj #2192. 「SHOI2014」概率充电器 题目描述 著名的电子产品品牌 SHOI 刚刚发布了引领世界潮流的下一代电子产品--概率充电器: 「采用全新纳米级加工技术,实现元件与导线能否通电完 ...

  7. Loj #3096. 「SNOI2019」数论

    Loj #3096. 「SNOI2019」数论 题目描述 给出正整数 \(P, Q, T\),大小为 \(n\) 的整数集 \(A\) 和大小为 \(m\) 的整数集 \(B\),请你求出: \[ \ ...

  8. Loj #3093. 「BJOI2019」光线

    Loj #3093. 「BJOI2019」光线 题目描述 当一束光打到一层玻璃上时,有一定比例的光会穿过这层玻璃,一定比例的光会被反射回去,剩下的光被玻璃吸收. 设对于任意 \(x\),有 \(x\t ...

  9. Loj #3089. 「BJOI2019」奥术神杖

    Loj #3089. 「BJOI2019」奥术神杖 题目描述 Bezorath 大陆抵抗地灾军团入侵的战争进入了僵持的阶段,世世代代生活在 Bezorath 这片大陆的精灵们开始寻找远古时代诸神遗留的 ...

随机推荐

  1. 洛谷P2633 Count on a tree(主席树,倍增LCA,树上差分)

    洛谷题目传送门 题目大意 就是给你一棵树,每个点都有点权,每次任意询问两点间路径上点权第k小的值(强制在线). 思路分析 第k小......又是主席树了.但这次变成树了,无法直接维护前缀和. 又是树上 ...

  2. 【题解】 [ZJOI2007]矩阵游戏 (二分图匹配)

    原题目戳我 Solution: 这个二分图藏还是挺深的,重点在哪里呢?首先我们分析下,交换影响的会是哪里. 每一次交换只会影响某一行上的排列或者某一列上的排列,如果有矩阵是下面这样,就一定不会互相影响 ...

  3. 解题:CF1118F2 Tree Cutting (Hard Version)

    题面 好题不问Div(这是Div3最后一题,不得不说Mike真是强=.=) 首先同一个颜色的点的LCA要和它们在一个划分出的块里,那么我们先按颜色把所有点到它们的LCA的路径涂色,如果这个过程中出现了 ...

  4. CF1027C Minimum Value Rectangle

    之前做的时候没想出来...现在来数学推导一波. 题意:从n个木棒中选出4个拼成一个矩形,使得 (周长)2/面积 最小. 解:设矩形宽a长b.我们要最小化下面这个式子: 去掉常数,不妨设b = a + ...

  5. activiti复盘重推的一种简单实现方式:

    activiti复盘重推的一种简单实现方式: 设置流程的每一步让用户选择,比如一共有6步完成,用户选择从第4步开始复盘重推,那么把原来的推演oldId和4传到后台, 首先,后台生成一个新的推演id n ...

  6. 为什么使用消息队列,为什么使用RabbitMQ、springAMQP

    1.为什么使用消息队列? 2.为什么使用RabbbitMQ? 3.为什么使用spring AMQP?

  7. android 系统开发板挂载U盘

    cat /proc/partitions 查看有u盘设备 df 查看挂载情况 iTOP4416开发板插入u盘,自动挂载到 /mnt/udisk1

  8. xgboost入门与实战(原理篇)

    sklearn实战-乳腺癌细胞数据挖掘 https://study.163.com/course/introduction.htm?courseId=1005269003&utm_campai ...

  9. MySQL数据类型以及基本使用详解

    MySQL数据类型以及基本使用详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.MySQL服务器的主要组件 我们知道MySQL的主要组件主要是由服务端(mysqld)和客户端 ...

  10. .net MVC入门

    这里面之所以没有Sql语句但是也可以对数据库进行数据操作的原因就是Entity Framework.Entity Framework有三种模式,这里用的是Models模式. 网上有太多的.net MV ...