热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

WeighttheTree(树形dp)

题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&#

题目Link
题目学习link1
题目学习link2
题目学习link3
%%% 受益匪浅!
--------------------------------

通过了解题意可知
当 n = 2 时 为一种特殊情况,特判一下(样例温和捏~
当 n > 2 时 当一个节点为good节点时,与其相连的节点一定为非good点
这样模型就显现出来了 类似于link2中所讲树的最小点覆盖问题
这里用 0 表示 该点不是good点 , 1 表示为good点
且只有 0-0 , 0- 1 两种情况,dp即可求出最大的 good 点个数
由于还要求最小的节点权值和,开个结构体用dp数组一起求
状态转移见代码

/**@author:bzdhxs*@date:2022/03/15*@URL:
*/

#include
#include
#include
#include
#include
#include
#include
#include
#include
using namespace std;
template <typename T>
inline void read(T &s){s &#61; 0;T w &#61; 1, ch &#61; getchar();while (!isdigit(ch)) { if (ch &#61;&#61; &#39;-&#39;) w &#61; -1; ch &#61; getchar(); }while (isdigit(ch)) { s &#61; (s << 1) &#43; (s << 3) &#43; (ch ^48); ch &#61; getchar();} s *&#61; w;}
template <typename T>
inline void write(T s){if (s < 0) putchar(&#39;-&#39;), s &#61; -s;if (s > 9) write(s / 10);putchar(s % 10 &#43; &#39;0&#39;);}
#define int long long
#define _orz ios::sync_with_stdio(false),cin.tie(0)
#define mem(str,num) memset(str,num,sizeof(str))
#define forr(i,a,b) for(int i &#61; a; i <&#61; b;i&#43;&#43;)
#define forn(i,n) for(int i &#61; 0; i
#define dbg() cout <<"0k!"<
typedef long long ll;
int pri[16] &#61; {2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53};
const int inf &#61; 0x3f3f3f3f;
const int INF &#61; ~0ULL;
const int N &#61; 1e6&#43;10;int n;
vector<int> g[200005];
struct node{int g,s; // g 记录好点个数 s 记录权值和//用来取较优的点bool operator>(const node&t)const{if(g !&#61; t.g) return g > t.g; //优先good点个数&#xff0c;相同取权值小的return s < t.s;}
}dp[200005][2];//树形dp
void dfs1(int u,int fa){dp[u][0].s &#61; 1;// 非good点权值为1&#xff0c;最小dp[u][1].s &#61; g[u].size();// 作为good点权值为直接连边的个数dp[u][1].g &#61; 1;// 为good点故为1for(auto v:g[u]){if(v &#61;&#61; fa) continue;dfs1(v,u);// u作为good点dp[u][1].g &#43;&#61; dp[v][0].g; // 直接连边的均为非good点取dp[v][0]dp[u][1].s &#43;&#61; dp[v][0].s;// 当 u为非good点的时候&#xff0c;v可以为good 也可以为非good 取最优&#xff1b;int best &#61; (dp[v][0] > dp[v][1])?0:1;dp[u][0].g &#43;&#61; dp[v][best].g;dp[u][0].s &#43;&#61; dp[v][best].s;}
}int w[200005];// 再次遍历更新点的权值
void dfs2(int u,int fa,int bes){// good 就取相连点个数 否则取1&#xff1b;w[u] &#61; (bes)?g[u].size():1;for(auto v:g[u]){if(v &#61;&#61; fa) continue;// 当前节点为good那么相邻节点就为非goodif(bes) dfs2(v,u,0);else{int best &#61; (dp[v][0] > dp[v][1])?0:1;dfs2(v,u,best);}}
}
signed main()
{cin >> n;forr(i,1,n-1){int l,r;cin >> l >> r;g[l].push_back(r);g[r].push_back(l);}//特判2if(n &#61;&#61; 2){cout << 2 <<" " << 2 << endl;cout << 1 <<" "<< 1 << endl;return 0;}dfs1(1,0);int best &#61; (dp[1][0] > dp[1][1])?0:1;//求每个点的权值dfs2(1,0,best);// 输出1节点最优情况cout << dp[1][best].g <<" "<<dp[1][best].s<<endl;forr(i,1,n) cout << w[i] <<" \n"[i&#61;&#61;n];return 0;
}

推荐阅读
  • 本题涉及一棵由N个节点组成的树(共有N-1条边),初始时所有节点均为白色。题目要求处理两种操作:一是改变某个节点的颜色(从白变黑或从黑变白);二是查询从根节点到指定节点路径上的第一个黑色节点,若无则输出-1。 ... [详细]
  • Codeforces Round #566 (Div. 2) A~F个人题解
    Dashboard-CodeforcesRound#566(Div.2)-CodeforcesA.FillingShapes题意:给你一个的表格,你 ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本题探讨如何通过最大流算法解决农场排水系统的设计问题。题目要求计算从水源点到汇合点的最大水流速率,使用经典的EK(Edmonds-Karp)和Dinic算法进行求解。 ... [详细]
  • 本题探讨了一种字符串变换方法,旨在判断两个给定的字符串是否可以通过特定的字母替换和位置交换操作相互转换。核心在于找到这些变换中的不变量,从而确定转换的可能性。 ... [详细]
  • 本文探讨了 C++ 中普通数组和标准库类型 vector 的初始化方法。普通数组具有固定长度,而 vector 是一种可扩展的容器,允许动态调整大小。文章详细介绍了不同初始化方式及其应用场景,并提供了代码示例以加深理解。 ... [详细]
  • 本实验主要探讨了二叉排序树(BST)的基本操作,包括创建、查找和删除节点。通过具体实例和代码实现,详细介绍了如何使用递归和非递归方法进行关键字查找,并展示了删除特定节点后的树结构变化。 ... [详细]
  • 本文详细探讨了VxWorks操作系统中双向链表和环形缓冲区的实现原理及使用方法,通过具体示例代码加深理解。 ... [详细]
  • 本文详细介绍了 Java 中的 org.apache.hadoop.registry.client.impl.zk.ZKPathDumper 类,提供了丰富的代码示例和使用指南。通过这些示例,读者可以更好地理解如何在实际项目中利用 ZKPathDumper 类进行注册表树的转储操作。 ... [详细]
  • This document outlines the recommended naming conventions for HTML attributes in Fast Components, focusing on readability and consistency with existing standards. ... [详细]
  • 本文详细介绍了C语言中链表的两种动态创建方法——头插法和尾插法,包括具体的实现代码和运行示例。通过这些内容,读者可以更好地理解和掌握链表的基本操作。 ... [详细]
  • Linux设备驱动程序:异步时间操作与调度机制
    本文介绍了Linux内核中的几种异步延迟操作方法,包括内核定时器、tasklet机制和工作队列。这些机制允许在未来的某个时间点执行任务,而无需阻塞当前线程,从而提高系统的响应性和效率。 ... [详细]
  • 本文探讨了《魔兽世界》中红蓝两方阵营在备战阶段的策略与实现方法,通过代码展示了双方如何根据资源和兵种特性进行战士生产。 ... [详细]
  • 本题通过将每个矩形视为一个节点,根据其相对位置构建拓扑图,并利用深度优先搜索(DFS)或状态压缩动态规划(DP)求解最小涂色次数。本文详细解析了该问题的建模思路与算法实现。 ... [详细]
author-avatar
俊维肇民74
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有