动态规划:P2585[ZJOI2006]三色二叉树

P2585[ZJOI2006]三色二叉树
题目:
思路分析:
可以借鉴动态规划:P1352没有上司的舞会 - 朱朱成 - 博客园 (cnblogs.com),建立一个二维的dp 第二维01代表有没有去,有没有染成绿色的。对于一个结点来说,以这个结点为根的子树,染成绿色的数目,因为是二叉树,所以一个结点只会连两个结点,所以不染成绿色的就等于max(左子节点染成绿色,的左子树绿色数目总和,右子节点不染成绿色,的右子树绿色数目总和),因为不染成绿色,左右子节点必有一个染成绿色,取最优就行,因为是三种颜色,所以不可能父节点不成绿色,两个子节点也不是绿色,可以根据抽屉原理得知。然后其他的计算其实跟其他两种颜色没关系了,就考虑染不染成绿色。然后我们再计算这个点染成绿色,就等于左子节点不染成绿色,的左子树绿色数目总和+右子节点不染成绿色,的右子树绿色数目总和+1。注意+1,就是父节点刚染成绿色的1。然后利用DFS,按照这个思路树形DP即可。因为题目是求最多和最少的,所以构建两个DP数组,max改成min即可。
关键DP代码:
  

 

 

 

 

AC完整代码:
  
 1 #include <cstring>
 2 #include <cmath>
 3 #include <cstdio>
 4 #include<iostream>
 5 using namespace std;
 6 const int maxn = 5 * 1e5 + 5;
 7 //设tree[i][0] 为第i个节点的左节点编号
 8 //设tree[i][1] 为第i个节点的右节点编号
 9 //dp[i][0]为第i个节点 为根 并且这个结点涂绿色 的子树 绿色最多的数量
10 //dp[i][12]为第i个节点 为根 并且这个结点不涂绿色 的子树 绿色最多的数量
11 int dp1[maxn][2];//一个求max 一个求min
12 int dp2[maxn][2];
13 int tree[maxn][2];
14 int cnt;
15 string str;
16 void dfs(int root)
17 {
18     cnt++;//表示访问了x个点
19     if (str[root] == '0')
20         return;
21     if (str[root] == '1')
22     {
23         tree[root][0] = root + 1;//如果有只有一个子节点 一定是左边结点 并且就是下一个root+1
24         dfs(root + 1);//搜索下一个点
25     }
26     else if (str[root] == '2')
27     {
28         tree[root][0] = root + 1;
29         dfs(root + 1);
30         tree[root][1] = cnt + 1;//右结点就是访问的点数+1
31         dfs(cnt + 1);
32     }
33 }
34 
35 void dp_dfs1(int x)
36 {
37     if (x == 0)//如果只有一个子节点的点 他的邻接点就是默认为0 搜到这些点直接return
38     {
39         return;
40     }
41     if (str[x] == '0')//如果是叶子
42     {
43         dp1[x][0] = 1;
44         return;
45     }
46     else
47     {
48         dp_dfs1(tree[x][0]);
49         dp_dfs1(tree[x][1]);
50         dp1[x][0] = dp1[tree[x][0]][1] + dp1[tree[x][1]][1]+1;//结点染绿色就等于左右两边不染的颜色+1
51         //cout << x << " " << "左边" << ":" << dp[x][0] << " " << endl;
52         dp1[x][1] = max(dp1[tree[x][0]][0] + dp1[tree[x][1]][1], dp1[tree[x][0]][1] + dp1[tree[x][1]][0]);
53         //cout << x << " " << "右边" << ":" << dp[x][1] << " " << endl;
54     }
55 }
56 void dp_dfs2(int x)
57 {
58     if (x == 0)//如果只有一个子节点的点 他的邻接点就是默认为0 搜到这些点直接return
59     {
60         return;
61     }
62     if (str[x] == '0')//如果是叶子
63     {
64         dp2[x][0] = 1;
65         return;
66     }
67     else
68     {
69         dp_dfs2(tree[x][0]);
70         dp_dfs2(tree[x][1]);
71         dp2[x][0] = dp2[tree[x][0]][1] + dp2[tree[x][1]][1] + 1;//结点染绿色就等于左右两边不染的颜色+1
72         //cout << x << " " << "左边" << ":" << dp[x][0] << " " << endl;
73         dp2[x][1] = min(dp2[tree[x][0]][0] + dp2[tree[x][1]][1], dp2[tree[x][0]][1] + dp2[tree[x][1]][0]);
74         //cout << x << " " << "右边" << ":" << dp[x][1] << " " << endl;
75     }
76 }
77 int main()
78 {
79     cin >> str;
80     for (int i = str.length(); i >= 1; --i)
81     {
82         str[i] = str[i - 1];
83     }
84     dfs(1);
85     dp_dfs1(1);
86     dp_dfs2(1);
87     cout << max(dp1[1][0], dp1[1][1]);
88     cout << " ";
89     cout << min(dp2[1][0], dp2[1][1]);
90     return 0;
91 
92 }

 

 

posted @ 2022-05-08 11:21  朱朱成  阅读(56)  评论(0编辑  收藏  举报