解析:
开始认为分情况讨论 k 小于等于叶子结点和大于叶子结点的情况,然后选择深度最深的叶子结点和子孙数量最小的结点,但是发现如果把某一个非叶子结点选取,那么其子孙的贡献都会减少。
考虑贪心,首先DFS出每个节点的深度deep(根节点为 0 )和每个节点的子孙结点个数 num(不带本身),这样如果某个结点被选取,那么其贡献为 deep - num ,所以我们选取最大的 k 个结点累计即可。
此处贪心的正确性证明:如果我们要选择某个结点,那么他的所有子孙结点肯定要被选择。如果不这样的话,那么显然选取他的子孙结点对于答案的贡献更高(deep更大,num更小),所以此时这个结点的子孙结点肯定都被选择,所以贡献值为 deep - num
- #include
- using namespace std;
- #define int long long
- const int N=2e5+5;
- int n,k,dis[N];
- vector<int>e[N];
- priority_queue<int>q;
- int dfs(int u,int deep,int fa){
- dis[u]=deep;
- if(e[u].size()==1&&u!=1){ //叶结点
- q.push(dis[u]);
- return 1;
- }
- int cnt=0;
- for(int i=0;i
size();i++){ - if(e[u][i]!=fa) cnt+=dfs(e[u][i],deep+1,u);
- }
- q.push(dis[u]-cnt); //优先队列统计
- return cnt+1; //返回子孙结点个数
- }
- signed main(){
- scanf("%lld%lld",&n,&k);
- for(int i=1;i
- int a,b;
- scanf("%lld%lld",&a,&b);
- e[a].push_back(b);
- e[b].push_back(a);
- }
- dfs(1,0,-1);
- int res=0;
- while(k&&q.size()){
- res+=q.top();
- q.pop();
- k--;
- }
- cout<
- return 0;
- }
-
相关阅读:
拿什么拯救你?这暴热的天气!只能在家用python分析做个可视化康康
踩坑npm install qrcodejs2和crypto-js
Jetson nano嵌入式平台配置ip记录
问界M7的诸多优点(自动驾驶走进我们的生活二)
电力系统直流潮流分析【N-1】(Matlab代码实现)
西安邮电大学第三届网络安全技能大赛---PWN方向WP
如何理解volatile关键字?
【GIS】地理坐标系与投影坐标系的区别
实际经历学来的管理知识(1)
1111 Online Map
-
原文地址:https://blog.csdn.net/JungleZRD/article/details/133779601