Codeforces 1153D Serval and Rooted Tree (简单树形DP)
<题目链接>
题目大意:
Serval拥有的有根树有n个节点,节点1是根。 Serval会将一些数字写入树的所有节点。但是,有一些限制。除叶子之外的每个节点都有一个写入操作的最大值或最小值,表示该节点中的数字应分别等于其子节点中所有数字的最大值或最小值。
假设树中有k个叶子。 Serval希望将整数1,2,...,k放到k个叶子上(每个数字应该只使用一次)。他喜欢大的数字,因此他希望最大化根的数字。作为他最好的朋友,你能帮助他吗?
解题分析:
不难想到,本题就是转化成求在满足题目求的情况下,传递到根节点的最少叶子节点个数。因为叶子节点的值是由我们自己够构造,所以我们只需要将那些能够传递到根节点的叶子节点,从大到小赋值,就能够让根节点的值最大,且它的最大值为ans-num+1(ans为叶子节点个数,num为传递到根节点的叶子节点个数)。
#include <bits/stdc++.h> using namespace std; template<typename T> inline void read(T&x){ x=0;int f=1;char c=getchar(); while(c<'0' || c>'9'){ if(c=='-')f=-1;c=getchar(); } while(c>='0' && c<='9'){ x=x*10+c-'0';c=getchar(); } x*=f; } const int N = 3e5+5; int val[N],num[N],ans=0; vector<int>G[N]; void dfs(int u){ if(G[u].size()==0){ num[u]=1;ans++;return; } int tmp=0; for(int i=0;i<G[u].size();i++){ int v=G[u][i]; dfs(v); tmp+=num[v]; //得到所有子树的所有叶子节点个数 if(val[u]==1)num[u]=min(num[u],num[v]); //如果是max的话,要使传递到根节点的叶子个数最少,只需要选某颗子树中最少的叶子个数 else num[u]=max(num[u],tmp); //如果是min的话,别无选择,只能选子树中叶子节点最多的值 } } int main(){ int n;read(n); for(int i=1;i<=n;i++){ read(val[i]); if(val[i]==1)num[i]=1e9; else num[i]=0; } for(int i=2;i<=n;i++){ int u;read(u); G[u].push_back(i); } dfs(1); printf("%d\n",ans-num[1]+1); }
作者:is_ok
出处:http://www.cnblogs.com/00isok/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。
分类:
图论———树形DP
, DP———@树形DP
标签:
树形DP
【推荐】还在用 ECharts 开发大屏?试试这款永久免费的开源 BI 工具!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从二进制到误差:逐行拆解C语言浮点运算中的4008175468544之谜
· .NET制作智能桌面机器人:结合BotSharp智能体框架开发语音交互
· 软件产品开发中常见的10个问题及处理方法
· .NET 原生驾驭 AI 新基建实战系列:向量数据库的应用与畅想
· 从问题排查到源码分析:ActiveMQ消费端频繁日志刷屏的秘密
· C# 13 中的新增功能实操
· Ollama本地部署大模型总结
· 【杭电多校比赛记录】2025“钉耙编程”中国大学生算法设计春季联赛(4)
· langchain0.3教程:从0到1打造一个智能聊天机器人
· 2025成都.NET开发者Connect圆满结束
2018-04-14 UVA 129困难的串【DFS】