\(\text{Solution}\)

设 \(f_{x,i}\) 表示以 \(x\) 为根的子树 \(i\) 天后的最大收益

那么 \(f_{x,i} = \max(f_{x,i-1},w_x [d[x] \ge i] + \sum f_{v,i})\)

这样的转移时 \(O(nk)\) 的,只能拿到 \(34pts\)

再加上其他的部分分也只有 \(48pts\)

正解是在这个 \(dp\) 上优化

我们发现很多状态都是没用的

观察到 \(x\) 各个状态的值由 \(v\) 对应的状态合并上来

那么就可以考虑线段树合并

开 \(n\) 棵线段树代替 \(f\) 的第二维,也就是时间这一维

转移就成线段树的合并操作了

但发现 \(f\) 的值是前缀最大值,需要我们获得前缀最大值之后再合并(相加)才对

那么线段树合并的时候维护两棵线段树前缀最大值,给对方

合并完子树后考虑加入当前节点的贡献

\(x\) 会对它的线段树 \(d_x\) 以后的值产生影响

并且此时还需取出 \(d_x\) 前的最大值加上 \(w_x\) 才是正确的贡献

对 \(d_x\) 以后的值取 \(max\) 即可

\(\text{Code}\)

#include <cstdio>
#include <iostream>
#define re register
using namespace std;
typedef long long LL; const int N = 1e5 + 5;
int n, m, k, fa[N], tot, h[N];
struct node{int d, w;}a[N];
struct edge{int to, nxt;}e[N];
inline void add(int x, int y){e[++tot] = edge{y, h[x]}, h[x] = tot;} inline void read(int &x)
{
x = 0; char ch = getchar();
while (ch < '0' || ch > '9') ch = getchar();
while (ch >= '0' && ch <= '9') x = (x << 3) + (x << 1) + ch - '0', ch = getchar();
} int size, ls[N << 5], rs[N << 5], rt[N];
LL val[N << 5], tag[N << 5]; inline void pushdown(int p)
{
if (!tag[p]) return;
if (ls[p]) val[ls[p]] += tag[p], tag[ls[p]] += tag[p];
if (rs[p]) val[rs[p]] += tag[p], tag[rs[p]] += tag[p];
tag[p] = 0;
} void update(int &p, int l, int r, int x, LL v)
{
if (!p) p = ++size, ls[p] = rs[p] = val[p] = tag[p] = 0;
val[p] = max(val[p], v);
if (l == r) return;
pushdown(p);
int mid = (l + r) >> 1;
if (x <= mid) update(ls[p], l, mid, x, v);
else update(rs[p], mid + 1, r, x, v);
} LL query(int p, int l, int r, int x)
{
if (!p || r <= x) return val[p];
pushdown(p);
int mid = (l + r) >> 1; LL res = query(ls[p], l, mid, x);
if (x > mid) res = max(res, query(rs[p], mid + 1, r, x));
return res;
} int merge(int x, int y, LL tagx, LL tagy, int l, int r)
{
if (!x && !y) return 0;
if (!y){val[x] += tagx, tag[x] += tagx; return x;}
if (!x){val[y] += tagy, tag[y] += tagy; return y;}
if (l == r)
{
tagx = max(tagx, val[y]), tagy = max(tagy, val[x]);
val[x] = max(val[x] + tagx, val[y] + tagy);
return x;
}
pushdown(x), pushdown(y);
int mid = (l + r) >> 1;
LL valx = val[ls[x]], valy = val[ls[y]];
ls[x] = merge(ls[x], ls[y], tagx, tagy, l, mid);
rs[x] = merge(rs[x], rs[y], max(tagx, valy), max(tagy, valx), mid + 1, r);
val[x] = max(val[ls[x]], val[rs[x]]);
return x;
} void dfs(int x)
{
for(re int i = h[x]; i; i = e[i].nxt)
dfs(e[i].to), rt[x] = merge(rt[x], rt[e[i].to], 0, 0, 1, k);
if (a[x].d) update(rt[x], 1, k, a[x].d, query(rt[x], 1, k, a[x].d) + a[x].w);
} int main()
{
read(n), read(m), read(k);
for(re int i = 2; i <= n; i++) read(fa[i]), add(fa[i], i);
for(re int i = 1, v; i <= m; i++) read(v), read(a[v].d), read(a[v].w);
dfs(1), printf("%lld\n", val[rt[1]]);
}

「CEOI2019」魔法树的更多相关文章

  1. liberOJ #2033. 「SDOI2016」生成魔咒 后缀数组

    #2033. 「SDOI2016」生成魔咒     题目描述 魔咒串由许多魔咒字符组成,魔咒字符可以用数字表示.例如可以将魔咒字符 1 11.2 22 拼凑起来形成一个魔咒串 [1,2] [1, 2] ...

  2. 「译」JUnit 5 系列:条件测试

    原文地址:http://blog.codefx.org/libraries/junit-5-conditions/ 原文日期:08, May, 2016 译文首发:Linesh 的博客:「译」JUni ...

  3. 「译」JUnit 5 系列:扩展模型(Extension Model)

    原文地址:http://blog.codefx.org/design/architecture/junit-5-extension-model/ 原文日期:11, Apr, 2016 译文首发:Lin ...

  4. JavaScript OOP 之「创建对象」

    工厂模式 工厂模式是软件工程领域一种广为人知的设计模式,这种模式抽象了创建具体对象的过程.工厂模式虽然解决了创建多个相似对象的问题,但却没有解决对象识别的问题. function createPers ...

  5. 「C++」理解智能指针

    维基百科上面对于「智能指针」是这样描述的: 智能指针(英语:Smart pointer)是一种抽象的数据类型.在程序设计中,它通常是经由类型模板(class template)来实做,借由模板(tem ...

  6. 「JavaScript」四种跨域方式详解

    超详细并且带 Demo 的 JavaScript 跨域指南来了! 本文基于你了解 JavaScript 的同源策略,并且了解使用跨域跨域的理由. 1. JSONP 首先要介绍的跨域方法必然是 JSON ...

  7. 「2014-5-31」Z-Stack - Modification of Zigbee Device Object for better network access management

    写一份赏心悦目的工程文档,是很困难的事情.若想写得完善,不仅得用对工具(use the right tools),注重文笔,还得投入大把时间,真心是一件难度颇高的事情.但,若是真写好了,也是善莫大焉: ...

  8. 「2014-3-18」multi-pattern string match using aho-corasick

    我是擅(倾)长(向)把一篇文章写成杂文的.毕竟,写博客记录生活点滴,比不得发 paper,要求字斟句酌八股结构到位:风格偏杂文一点,也是没人拒稿的.这么说来,arxiv 就好比是 paper 世界的博 ...

  9. 「2014-3-17」C pointer again …

    记录一个比较基础的东东-- C 语言的指针,一直让人又爱又恨,爱它的人觉得它既灵活又强大,恨它的人觉得它太过于灵活太过于强大以至于容易将人绕晕.最早接触 C 语言,还是在刚进入大学的时候,算起来有好些 ...

  10. 「2014-3-13」Javascript Engine, Java VM, Python interpreter, PyPy – a glance

    提要: url anchor (ajax) => javascript engine (1~4 articles) => java VM vs. python interpreter =& ...

随机推荐

  1. MIUI12解决安装charles抓包安装证书后还是提示证书不安全

    前言 我抓包这么长时间,这个问题我还是第一次遇到,导致我反复试验,明明安装证书还是提示不安全.我用新买的手机MIUI 12系统弄了半天 解决方案 首先下载证书这部分是一个坑,小米 MIUI 12系统下 ...

  2. 漫谈计算机网络:网络层 ------ 重点:IP协议与互联网路由选择协议

    面试答不上?计网很枯燥? 听说你学习 计网 每次记了都会忘? 不妨抽时间和我一起多学学它 深入浅出,用你的空闲时间来探索计算机网络的硬核知识! 博主的上篇连载文章<初识图像处理技术> 图像 ...

  3. Django连接数据库(mysql)与Django ORM实操(增删改查) 前端页面

    目录 一:pycharm链接数据库(MySQL) 1.pycharm右侧Database工具栏 2.下载对应的驱动,填写MySQL数据库信息连接(当前为客户端) 3.pycharm连接MySQL数据库 ...

  4. 自定义RBAC(5)

    您好,我是湘王,这是我的博客园,欢迎您来,欢迎您再来- 把实体类及Service类都准备好了之后,就可以开始继续写业务代码了.Spring Security的强大之一就在于它的拦截器.那么这里也可以参 ...

  5. python Flask 操作数据库(2)

    单表操作 数据准备 from flask import Flask from flask_sqlalchemy import SQLAlchemy class Config: DEBUG = True ...

  6. 网络监测工具之Zabbix的搭建与测试方法(三) ---Zabbix Agent

    安装客户端 在官方网站下载最新版本zabbix agent v6.2.6,然后默认安装,其中配置服务端的界面如下图: 其他一律默认即可. 启用发现功能 如上图所示,开启发现规则,默认搜索zabbix ...

  7. 痞子衡嵌入式:探讨i.MXRT下FlexSPI driver实现Flash编程时对于中断支持问题

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT下FlexSPI driver实现Flash编程时对于中断支持问题. 前段时间有客户在官方社区反映 i.MXRT1170 下 ...

  8. [常用工具] PyAutoGUI使用教程

    PyAutoGUI使用教程 目录 PyAutoGUI使用教程 1 基础知识 2 一般函数 3 故障保险 4 鼠标函数 4.1 鼠标移动 4.2 鼠标拖动 4.3 鼠标单击 4.4 鼠标滚动 4.5 鼠 ...

  9. [python] ​python-pinyin库

    python-pinyin库是一个汉字拼音转换工具,其主要功能有: 根据词组智能匹配最正确的拼音. 支持多音字. 简单的繁体支持, 注音支持. 支持多种不同拼音风格. 安装命令为:pip instal ...

  10. C语言常用知识总结

    在 C 语言中,常量是一种固定值的标识符,它的值在程序执行期间不会改变. C 语言中有几种不同类型的常量: 字符常量:用单引号括起来的单个字符,例如 'A'.'b'.'1' 等. 字符串常量:用双引号 ...