- 标签:位运算、树、动态规划、状态压缩、枚举
- 难度:困难
描述:给定一个整数
要求:返回一个大小为
说明:
- 两个城市间距离:定义为它们之间需要经过的边的数目。
- 一棵子树:城市的一个子集,且子集中任意城市之间可以通过子集中的其他城市和边到达。两个子树被认为不一样的条件是至少有一个城市在其中一棵子树中存在,但在另一棵子树中不存在。
-
$2 \le n \le 15$ 。 -
$edges.length == n - 1$ 。 -
$edges[i].length == 2$ 。 -
$1 \le u_i, v_i \le n$ 。 - 题目保证
$(ui, vi)$ 所表示的边互不相同。
示例:
- 示例 1:
输入:n = 4, edges = [[1,2],[2,3],[2,4]]
输出:[3,4,0]
解释:
子树 {1,2}, {2,3} 和 {2,4} 最大距离都是 1 。
子树 {1,2,3}, {1,2,4}, {2,3,4} 和 {1,2,3,4} 最大距离都为 2 。
不存在城市间最大距离为 3 的子树。
- 示例 2:
输入:n = 2, edges = [[1,2]]
输出:[1]
因为题目中给定
而对于一个确定的子树来说,求子树中两个城市间距离就是在求子树的直径,这就跟 「1245. 树的直径」 和 「2246. 相邻字符不同的最长路径」 一样了。
那么这道题的思路就变成了:
- 通过二进制枚举的方式,得到所有子树。
- 对于当前子树,通过树形 DP + 深度优先搜索的方式,计算出当前子树的直径。
- 统计所有子树直径中经过的不同边数个数,将其放入答案数组中。
class Solution:
def countSubgraphsForEachDiameter(self, n: int, edges: List[List[int]]) -> List[int]:
graph = [[] for _ in range(n)] # 建图
for u, v in edges:
graph[u - 1].append(v - 1)
graph[v - 1].append(u - 1)
def dfs(mask, u):
nonlocal visited, diameter
visited |= 1 << u # 标记 u 访问过
u_len = 0 # u 节点的最大路径长度
for v in graph[u]: # 遍历 u 节点的相邻节点
if (visited >> v) & 1 == 0 and mask >> v & 1: # v 没有访问过,且在子集中
v_len = dfs(mask, v) # 相邻节点的最大路径长度
diameter = max(diameter, u_len + v_len + 1) # 维护最大路径长度
u_len = max(u_len, v_len + 1) # 更新 u 节点的最大路径长度
return u_len
ans = [0 for _ in range(n - 1)]
for mask in range(3, 1 << n): # 二进制枚举子集
if mask & (mask - 1) == 0: # 子集至少需要两个点
continue
visited = 0
diameter = 0
u = mask.bit_length() - 1
dfs(mask, u) # 在子集 mask 中递归求树的直径
if visited == mask:
ans[diameter - 1] += 1
return ans
-
时间复杂度:$O(n \times 2^n)$,其中
$n$ 为给定的城市数目。 - 空间复杂度:$O(n)$。