牛客练习赛39 B:选点(二叉树遍历+LIS)
发布日期:2022-04-01 13:25:20 浏览次数:26 分类:博客文章

本文共 1187 字,大约阅读时间需要 3 分钟。

链接:

来源:牛客网

时间限制:C/C++ 1秒,其他语言2秒

空间限制:C/C++ 131072K,其他语言262144K
64bit IO Format: %lld

题目描述

有一棵n个节点的二叉树,1为根节点,每个节点有一个值

w
i
w_i
。现在要选出尽量多的点。

对于任意一棵子树,都要满足:

如果选了根节点的话,在这棵子树内选的其他的点都要比根节点的值

如果在左子树选了一个点,在右子树中选的其他点要比它

输入描述:

第一行一个整数n。第二行n个整数

w
i
w_i
,表示每个点的权值。
接下来
n
n
行,每行两个整数
a
,
b
a,b
。第
i
+
2
i+2
行表示第
i
i
个节点的左右儿子节点。没有为
0
0
n
,
a
,
b
1
0
5
,
2
×
1
0
9
w
i
2
×
1
0
9
n,a,b≤10^5,−2×10^9≤w_i≤2×10^9

输出描述:

一行一个整数表示答案。

输入

51 5 4 2 33 24 50 00 00 0

输出

3

Solve

题目要求选出来的点满足:左儿子权值>右儿子权值>父亲权值。对二叉树进行后续遍历,可以得到左儿子->右儿子->父亲的排列,然后对遍历得到的结果反向求LIS即可

Code

#include 
#define ll long long#define ull unsigned long long#define ms(a,b) memset(a,b,sizeof(a))#define INF 0x7f7f7f7fconst int maxn=1e5+10;const int mod=1e9+7;using namespace std;int w[maxn];int arr[maxn];int ar[maxn];int dp[maxn];struct wzy{ int l,r;}p[maxn];int cnt;void get_arr(int x){ if(p[x].l) get_arr(p[x].l); if(p[x].r) get_arr(p[x].r); arr[cnt++]=w[x];}int main(int argc, char const *argv[]){ ios::sync_with_stdio(false); cin.tie(0); ms(p,0); int n,a,b; cin>>n; for(int i=1;i<=n;i++) cin>>w[i]; for(int i=1;i<=n;i++) { cin>>a>>b; p[i].l=a; p[i].r=b; } get_arr(1); for(int i=0;i

转载地址:https://www.cnblogs.com/Friends-A/p/11054971.html 如侵犯您的版权,请留言回复原文章的地址,我们会给您删除此文章,给您带来不便请您谅解!

上一篇:《机器学习实战》kNN算法及约会网站代码详解
下一篇:洛谷 P1439 【模板】最长公共子序列(DP,LIS?)

发表评论

最新留言

路过按个爪印,很不错,赞一个!
[***.219.124.196]2024年03月30日 17时37分27秒

关于作者

    喝酒易醉,品茶养心,人生如梦,品茶悟道,何以解忧?唯有杜康!
-- 愿君每日到此一游!

推荐文章