multiset 启发式合并贪心维护 LIS 的做法就不多说了,网上题解一大堆,着重讲一下线段树合并维护 \(dp\)。

\(O(n^2)\) 的 \(dp\) 非常显然。离散化后,设 \(dp[u][i]\) 表示节点 \(u\) 的子树中,最大值为 \(i\) 时最多取多少个节点。转移时考虑是否将节点 \(u\) 加入大根堆并分类讨论。

这样的状态不支持快速合并。考虑优化状态,设 \(dp[u][i]\) 表示节点 \(u\) 的子树中,最大值 \(\le i\) 时最多取多少个节点,并尝试使用线段树合并来维护。

转移同样考虑两种情况。如果 \(u\) 不取,那么直接 \(dp[u][i] = \sum dp[v][i]\) 即可,将 \(u\) 的儿子的线段树合并。合并之后,如果 \(u\) 取,就要求子树中取的最大值都 \(\lt val[u]\),那么要用 \(\max\limits_{i \lt val[u]}dp[u][i]+1\) 去更新 \(dp[u][\ge val[u]]\)。

注意到,根据状态的设计,\(dp[u]\) 其实是一个单调不减的序列,所以 \(\max\limits_{i \lt val[u]} dp[u][i]+1\) 其实相当于 \(dp[u][val[u]-1]+1\)。同时由于 \(dp[u]\) 单调不减,所以 \(dp[u][\ge val[u]]\) 中会被更新的值是一段左端点为 \(val[u]\) 的区间,这段区间的值都是 \(dp[u][val[u]-1]\)。

于是,可以二分出区间的右端点。具体地,二分找到最后一个 \(i\) 使得 \(dp[u][i] \lt dp[u][val[u]-1]+1\),然后在线段树上将区间 \([val[u],i]\) 覆盖或直接 \(+1\) 即可,需要标记永久化。

综上所述,该算法的时间复杂度为 \(O(n \log^2 n)\)。

/**
* @file: BZOJ4919.cpp
* @author: yaoxi-std
* @url:
*/
#pragma GCC optimize ("O2")
#pragma GCC optimize ("Ofast", "inline", "-ffast-math")
#pragma GCC target ("avx,sse2,sse3,sse4,mmx")
#include <bits/stdc++.h>
using namespace std;
#define resetIO(x) \
freopen(#x ".in", "r", stdin), freopen(#x ".out", "w", stdout)
#define debug(fmt, ...) \
fprintf(stderr, "[%s:%d] " fmt "\n", __FILE__, __LINE__, ##__VA_ARGS__)
template <class _Tp>
inline _Tp& read(_Tp& x) {
bool sign = false; char ch = getchar();
for (; !isdigit(ch); ch = getchar()) sign |= (ch == '-');
for (x = 0; isdigit(ch); ch = getchar()) x = x * 10 + (ch ^ 48);
return sign ? (x = -x) : x;
}
template <class _Tp>
inline void write(_Tp x) {
if (x < 0) putchar('-'), x = -x;
if (x > 9) write(x / 10);
putchar((x % 10) ^ 48);
}
bool m_be;
using ll = long long;
const int MAXN = 2e5 + 10;
const int INF = 0x3f3f3f3f;
int n, m, a[MAXN], fa[MAXN], val[MAXN];
vector<int> g[MAXN];
struct SegmentTree {
struct Node {
int ls, rs, sum;
} nd[MAXN * 25];
int tot, rt[MAXN];
int& operator[](int i) { return rt[i]; }
void update(int& i, int l, int r, int ql, int qr, int v) {
if (ql > qr) return;
if (!i) i = ++tot;
if (ql <= l && r <= qr) return void(nd[i].sum += v);
int mid = (l + r) >> 1;
if (ql <= mid) update(nd[i].ls, l, mid, ql, qr, v);
if (qr > mid) update(nd[i].rs, mid + 1, r, ql, qr, v);
}
int query(int i, int l, int r, int p) {
if (!i || !p) return 0;
if (l == r) return nd[i].sum;
int mid = (l + r) >> 1;
if (p <= mid) return nd[i].sum + query(nd[i].ls, l, mid, p);
return nd[i].sum + query(nd[i].rs, mid + 1, r, p);
}
void merge(int& x, int y, int l, int r) {
if (!x || !y) return void(x = x | y);
if (l == r) return void(nd[x].sum += nd[y].sum);
int mid = (l + r) >> 1;
merge(nd[x].ls, nd[y].ls, l, mid);
merge(nd[x].rs, nd[y].rs, mid + 1, r);
nd[x].sum += nd[y].sum;
}
} sgt;
void dfs(int u) {
for (auto v : g[u]) dfs(v), sgt.merge(sgt[u], sgt[v], 1, m);
int tmp = sgt.query(sgt[u], 1, m, a[u] - 1) + 1;
int l = a[u], r = m, pos = a[u] - 1;
while (l <= r) {
int mid = (l + r) >> 1;
if (sgt.query(sgt[u], 1, m, mid) < tmp)
l = mid + 1, pos = mid;
else
r = mid - 1;
}
sgt.update(sgt[u], 1, m, a[u], pos, 1);
}
bool m_ed;
signed main() {
read(n);
for (int i = 1; i <= n; ++i)
read(a[i]), read(fa[i]), val[i] = a[i], g[fa[i]].push_back(i);
sort(val + 1, val + n + 1), m = unique(val + 1, val + n + 1) - val - 1;
for (int i = 1; i <= n; ++i) a[i] = lower_bound(val + 1, val + m + 1, a[i]) - val;
dfs(1), write(sgt.query(1, 1, m, m)), putchar('\n');
return 0;
}

最新文章

  1. 【工匠大道】Git的使用总结
  2. WPF入门教程系列一——基础
  3. elastic search 配置问题
  4. C puzzles详解
  5. 《算法导论》习题解答 Chapter 22.1-6(求universal sink 通用汇点)
  6. spring中的aware接口
  7. android下升级软件介绍
  8. CTabCtrl
  9. DNS原理及其解析过程
  10. C#—泛型_推迟一切可以推迟的东西
  11. 40. leetcode 202. Happy Number
  12. js面试题知识点全解(一变量类型和计算)
  13. Java NIO之网络编程
  14. 【问题解决方案】查看Python安装了哪些库(pandas, matplotlib等等)
  15. Javascript模版引擎mustache.js简介
  16. Java Lambda基础——Function, Consumer, Predicate, Supplier, 及FunctionalInterface接口
  17. Python 常见时间处理
  18. Redis---事务和Wtach
  19. cf 1029D
  20. 【Go命令教程】9. go list

热门文章

  1. redis bitmap数据结构之java对等操作
  2. 构建Springboot项目、实现简单的输出功能、将项目打包成可以执行的JAR包(详细图解过程)
  3. 齐博x1万能数据统计之fun函数
  4. SpringBoot内置工具类,告别瞎写工具类了
  5. 现在入行Java真的还有出路吗?
  6. 驱动开发:内核LDE64引擎计算汇编长度
  7. Java 编码那些事(二)
  8. 第2-1-3章 docker-compose安装FastDFS,实现文件存储服务
  9. Https Webservice接口的免证书调用
  10. Mp3文件标签信息读取和写入(Kotlin)