|
| 1 | +--- |
| 2 | +comments: true |
| 3 | +difficulty: 中等 |
| 4 | +edit_url: https://github.com/doocs/leetcode/edit/main/solution/3200-3299/3249.Count%20the%20Number%20of%20Good%20Nodes/README.md |
| 5 | +--- |
| 6 | + |
| 7 | +<!-- problem:start --> |
| 8 | + |
| 9 | +# [3249. 统计好节点的数目](https://leetcode.cn/problems/count-the-number-of-good-nodes) |
| 10 | + |
| 11 | +[English Version](/solution/3200-3299/3249.Count%20the%20Number%20of%20Good%20Nodes/README_EN.md) |
| 12 | + |
| 13 | +## 题目描述 |
| 14 | + |
| 15 | +<!-- description:start --> |
| 16 | + |
| 17 | +<p>现有一棵 <strong>无向</strong> 树,树中包含 <code>n</code> 个节点,按从 <code>0</code> 到 <code>n - 1</code> 标记。树的根节点是节点 <code>0</code> 。给你一个长度为 <code>n - 1</code> 的二维整数数组 <code>edges</code>,其中 <code>edges[i] = [a<sub>i</sub>, b<sub>i</sub>]</code> 表示树中节点 <code>a<sub>i</sub></code> 与节点 <code>b<sub>i</sub></code> 之间存在一条边。</p> |
| 18 | + |
| 19 | +<p>如果一个节点的所有子节点为根的 <span data-keyword="subtree">子树</span> 包含的节点数相同,则认为该节点是一个 <strong>好节点</strong>。</p> |
| 20 | + |
| 21 | +<p>返回给定树中<strong> 好节点 </strong>的数量。</p> |
| 22 | + |
| 23 | +<p><strong>子树</strong> 指的是一个节点以及它所有后代节点构成的一棵树。</p> |
| 24 | + |
| 25 | +<p> </p> |
| 26 | + |
| 27 | +<p> </p> |
| 28 | + |
| 29 | +<p><strong class="example">示例 1:</strong></p> |
| 30 | + |
| 31 | +<div class="example-block"> |
| 32 | +<p><strong>输入:</strong><span class="example-io">edges = [[0,1],[0,2],[1,3],[1,4],[2,5],[2,6]]</span></p> |
| 33 | + |
| 34 | +<p><strong>输出:</strong><span class="example-io">7</span></p> |
| 35 | + |
| 36 | +<p><strong>说明:</strong></p> |
| 37 | +<img alt="" src="https://fastly.jsdelivr.net/gh/doocs/leetcode@main/solution/3200-3299/3249.Count%20the%20Number%20of%20Good%20Nodes/images/tree1.png" style="width: 360px; height: 158px;" /> |
| 38 | +<p>树的所有节点都是好节点。</p> |
| 39 | +</div> |
| 40 | + |
| 41 | +<p><strong class="example">示例 2:</strong></p> |
| 42 | + |
| 43 | +<div class="example-block"> |
| 44 | +<p><strong>输入:</strong><span class="example-io">edges = [[0,1],[1,2],[2,3],[3,4],[0,5],[1,6],[2,7],[3,8]]</span></p> |
| 45 | + |
| 46 | +<p><strong>输出:</strong><span class="example-io">6</span></p> |
| 47 | + |
| 48 | +<p><strong>说明:</strong></p> |
| 49 | +<img alt="" src="https://fastly.jsdelivr.net/gh/doocs/leetcode@main/solution/3200-3299/3249.Count%20the%20Number%20of%20Good%20Nodes/images/screenshot-2024-06-03-193552.png" style="width: 360px; height: 303px;" /> |
| 50 | +<p>树中有 6 个好节点。上图中已将这些节点着色。</p> |
| 51 | +</div> |
| 52 | + |
| 53 | +<p><strong class="example">示例 3:</strong></p> |
| 54 | + |
| 55 | +<div class="example-block"> |
| 56 | +<p><span class="example-io"><b>输入:</b>edges = [[0,1],[1,2],[1,3],[1,4],[0,5],[5,6],[6,7],[7,8],[0,9],[9,10],[9,12],[10,11]]</span></p> |
| 57 | + |
| 58 | +<p><span class="example-io"><b>输出:</b>12</span></p> |
| 59 | + |
| 60 | +<p><strong>解释:</strong></p> |
| 61 | +<img alt="" src="https://fastly.jsdelivr.net/gh/doocs/leetcode@main/solution/3200-3299/3249.Count%20the%20Number%20of%20Good%20Nodes/images/rob.jpg" style="width: 450px; height: 277px;" /> |
| 62 | +<p>除了节点 9 以外其他所有节点都是好节点。</p> |
| 63 | +</div> |
| 64 | + |
| 65 | +<p> </p> |
| 66 | + |
| 67 | +<p><strong>提示:</strong></p> |
| 68 | + |
| 69 | +<ul> |
| 70 | + <li><code>2 <= n <= 10<sup>5</sup></code></li> |
| 71 | + <li><code>edges.length == n - 1</code></li> |
| 72 | + <li><code>edges[i].length == 2</code></li> |
| 73 | + <li><code>0 <= a<sub>i</sub>, b<sub>i</sub> < n</code></li> |
| 74 | + <li>输入确保 <code>edges</code> 总表示一棵有效的树。</li> |
| 75 | +</ul> |
| 76 | + |
| 77 | +<!-- description:end --> |
| 78 | + |
| 79 | +## 解法 |
| 80 | + |
| 81 | +<!-- solution:start --> |
| 82 | + |
| 83 | +### 方法一:DFS |
| 84 | + |
| 85 | +我们先根据题目给定的边 $\textit{edges}$ 构建出树的邻接表 $\textit{g}$,其中 $\textit{g}[a]$ 表示节点 $a$ 的所有邻居节点。 |
| 86 | + |
| 87 | +然后,我们设计一个函数 $\textit{dfs}(a, \textit{fa})$,表示计算以节点 $a$ 为根的子树中的节点数,并累计好节点的数量。其中 $\textit{fa}$ 表示节点 $a$ 的父节点。 |
| 88 | + |
| 89 | +函数 $\textit{dfs}(a, \textit{fa})$ 的执行过程如下: |
| 90 | + |
| 91 | +1. 初始化变量 $\textit{pre} = -1$, $\textit{cnt} = 1$, $\textit{ok} = 1$,分别表示节点 $a$ 的某个子树的节点数、节点 $a$ 的所有子树的节点数、以及节点 $a$ 是否为好节点。 |
| 92 | +2. 遍历节点 $a$ 的所有邻居节点 $b$,如果 $b$ 不等于 $\textit{fa}$,则递归调用 $\textit{dfs}(b, a)$,返回值为 $\textit{cur}$,并累加到 $\textit{cnt}$ 中。如果 $\textit{pre} < 0$,则将 $\textit{cur}$ 赋值给 $\textit{pre}$;否则,如果 $\textit{pre}$ 不等于 $\textit{cur}$,说明节点 $a$ 的不同子树的节点数不同,将 $\textit{ok}$ 置为 $0$。 |
| 93 | +3. 最后,累加 $\textit{ok}$ 到答案中,并返回 $\textit{cnt}$。 |
| 94 | + |
| 95 | +在主函数中,我们调用 $\textit{dfs}(0, -1)$,最后返回答案。 |
| 96 | + |
| 97 | +时间复杂度 $O(n)$,空间复杂度 $O(n)$。其中 $n$ 表示节点的数量。 |
| 98 | + |
| 99 | +<!-- tabs:start --> |
| 100 | + |
| 101 | +#### Python3 |
| 102 | + |
| 103 | +```python |
| 104 | +class Solution: |
| 105 | + def countGoodNodes(self, edges: List[List[int]]) -> int: |
| 106 | + def dfs(a: int, fa: int) -> int: |
| 107 | + pre = -1 |
| 108 | + cnt = ok = 1 |
| 109 | + for b in g[a]: |
| 110 | + if b != fa: |
| 111 | + cur = dfs(b, a) |
| 112 | + cnt += cur |
| 113 | + if pre < 0: |
| 114 | + pre = cur |
| 115 | + elif pre != cur: |
| 116 | + ok = 0 |
| 117 | + nonlocal ans |
| 118 | + ans += ok |
| 119 | + return cnt |
| 120 | + |
| 121 | + g = defaultdict(list) |
| 122 | + for a, b in edges: |
| 123 | + g[a].append(b) |
| 124 | + g[b].append(a) |
| 125 | + ans = 0 |
| 126 | + dfs(0, -1) |
| 127 | + return ans |
| 128 | +``` |
| 129 | + |
| 130 | +#### Java |
| 131 | + |
| 132 | +```java |
| 133 | +class Solution { |
| 134 | + private int ans; |
| 135 | + private List<Integer>[] g; |
| 136 | + |
| 137 | + public int countGoodNodes(int[][] edges) { |
| 138 | + int n = edges.length + 1; |
| 139 | + g = new List[n]; |
| 140 | + Arrays.setAll(g, k -> new ArrayList<>()); |
| 141 | + for (var e : edges) { |
| 142 | + int a = e[0], b = e[1]; |
| 143 | + g[a].add(b); |
| 144 | + g[b].add(a); |
| 145 | + } |
| 146 | + dfs(0, -1); |
| 147 | + return ans; |
| 148 | + } |
| 149 | + |
| 150 | + private int dfs(int a, int fa) { |
| 151 | + int pre = -1, cnt = 1, ok = 1; |
| 152 | + for (int b : g[a]) { |
| 153 | + if (b != fa) { |
| 154 | + int cur = dfs(b, a); |
| 155 | + cnt += cur; |
| 156 | + if (pre < 0) { |
| 157 | + pre = cur; |
| 158 | + } else if (pre != cur) { |
| 159 | + ok = 0; |
| 160 | + } |
| 161 | + } |
| 162 | + } |
| 163 | + ans += ok; |
| 164 | + return cnt; |
| 165 | + } |
| 166 | +} |
| 167 | +``` |
| 168 | + |
| 169 | +#### C++ |
| 170 | + |
| 171 | +```cpp |
| 172 | +class Solution { |
| 173 | +public: |
| 174 | + int countGoodNodes(vector<vector<int>>& edges) { |
| 175 | + int n = edges.size() + 1; |
| 176 | + vector<int> g[n]; |
| 177 | + for (const auto& e : edges) { |
| 178 | + int a = e[0], b = e[1]; |
| 179 | + g[a].push_back(b); |
| 180 | + g[b].push_back(a); |
| 181 | + } |
| 182 | + int ans = 0; |
| 183 | + auto dfs = [&](auto&& dfs, int a, int fa) -> int { |
| 184 | + int pre = -1, cnt = 1, ok = 1; |
| 185 | + for (int b : g[a]) { |
| 186 | + if (b != fa) { |
| 187 | + int cur = dfs(dfs, b, a); |
| 188 | + cnt += cur; |
| 189 | + if (pre < 0) { |
| 190 | + pre = cur; |
| 191 | + } else if (pre != cur) { |
| 192 | + ok = 0; |
| 193 | + } |
| 194 | + } |
| 195 | + } |
| 196 | + ans += ok; |
| 197 | + return cnt; |
| 198 | + }; |
| 199 | + dfs(dfs, 0, -1); |
| 200 | + return ans; |
| 201 | + } |
| 202 | +}; |
| 203 | +``` |
| 204 | +
|
| 205 | +#### Go |
| 206 | +
|
| 207 | +```go |
| 208 | +func countGoodNodes(edges [][]int) (ans int) { |
| 209 | + n := len(edges) + 1 |
| 210 | + g := make([][]int, n) |
| 211 | + for _, e := range edges { |
| 212 | + a, b := e[0], e[1] |
| 213 | + g[a] = append(g[a], b) |
| 214 | + g[b] = append(g[b], a) |
| 215 | + } |
| 216 | + var dfs func(int, int) int |
| 217 | + dfs = func(a, fa int) int { |
| 218 | + pre, cnt, ok := -1, 1, 1 |
| 219 | + for _, b := range g[a] { |
| 220 | + if b != fa { |
| 221 | + cur := dfs(b, a) |
| 222 | + cnt += cur |
| 223 | + if pre < 0 { |
| 224 | + pre = cur |
| 225 | + } else if pre != cur { |
| 226 | + ok = 0 |
| 227 | + } |
| 228 | + } |
| 229 | + } |
| 230 | + ans += ok |
| 231 | + return cnt |
| 232 | + } |
| 233 | + dfs(0, -1) |
| 234 | + return |
| 235 | +} |
| 236 | +``` |
| 237 | + |
| 238 | +#### TypeScript |
| 239 | + |
| 240 | +```ts |
| 241 | +function countGoodNodes(edges: number[][]): number { |
| 242 | + const n = edges.length + 1; |
| 243 | + const g: number[][] = Array.from({ length: n }, () => []); |
| 244 | + for (const [a, b] of edges) { |
| 245 | + g[a].push(b); |
| 246 | + g[b].push(a); |
| 247 | + } |
| 248 | + let ans = 0; |
| 249 | + const dfs = (a: number, fa: number): number => { |
| 250 | + let [pre, cnt, ok] = [-1, 1, 1]; |
| 251 | + for (const b of g[a]) { |
| 252 | + if (b !== fa) { |
| 253 | + const cur = dfs(b, a); |
| 254 | + cnt += cur; |
| 255 | + if (pre < 0) { |
| 256 | + pre = cur; |
| 257 | + } else if (pre !== cur) { |
| 258 | + ok = 0; |
| 259 | + } |
| 260 | + } |
| 261 | + } |
| 262 | + ans += ok; |
| 263 | + return cnt; |
| 264 | + }; |
| 265 | + dfs(0, -1); |
| 266 | + return ans; |
| 267 | +} |
| 268 | +``` |
| 269 | + |
| 270 | +<!-- tabs:end --> |
| 271 | + |
| 272 | +<!-- solution:end --> |
| 273 | + |
| 274 | +<!-- problem:end --> |
0 commit comments