leetcode每日一题之最小高度树

310、最小高度树

树是一个无向图,其中任何两个顶点只通过一条路径连接。 换句话说,一个任何没有简单环路的连通图都是一棵树。

给你一棵包含 n 个节点的树,标记为 0 到 n - 1 。给定数字 n 和一个有 n - 1 条无向边的 edges 列表(每一个边都是一对标签),其中 edges[i] = [ai, bi] 表示树中节点 ai 和 bi 之间存在一条无向边。

可选择树中任何一个节点作为根。当选择节点 x 作为根节点时,设结果树的高度为 h 。在所有可能的树中,具有最小高度的树(即,min(h))被称为 最小高度树 。

请你找到所有的 最小高度树 并按 任意顺序 返回它们的根节点标签列表。

树的 高度 是指根节点和叶子节点之间最长向下路径上边的数量。

示例1:

输入:n = 4, edges = [[1,0],[1,2],[1,3]]
输出:[1]
解释:如图所示,当根是标签为 1 的节点时,树的高度是 1 ,这是唯一的最小高度树。

示例2:

输入:n = 6, edges = [[3,0],[3,1],[3,2],[3,4],[5,4]]
输出:[3,4]

思路:

如果一棵树,能找到一条最长的路径,那么最小高度树的高度应该为这条路径的一半,而且根节点是这条路径的中间节点。那么这个问题就可以转换为找edges中的最长路径问题。

class Solution:
    def findMinHeightTrees(self, n: int, edges: List[List[int]]) -> List[int]:
        if n == 1: return [0]
				# 存储每两个点之间的线段
        ref = [[] for _ in range(n)]
        for x, y in edges:
            ref[x].append(y)
            ref[y].append(x)
        parent = [0] * n
				# 从起始节点遍历,记录已经遍历过的节点
        # 使用广度优先遍历,记录子节点和父节点的关系。
        def dfs(start):
            visited = [0] * n
            visited[start] = 1
            queue = [start]
            nxt = start
            while queue:
                nxt = queue.pop(0)
                for node in ref[nxt]:
                    if visited[node]: continue
                    parent[node] = nxt
                    visited[node] = 1
                    queue.append(node)
            return nxt
				# 首先从0找到最远距离的x,然后找到x最远距离的y,那么x->y就是树中最长距离
        x = dfs(0)
        y = dfs(x)
        # 根据记录的父子节点关系,倒推整个path路径
        path = []
        parent[x] = -1
        while y != -1:
            path.append(y)
            y = parent[y]
        m = len(path)
        # 如果path是奇数,那么根节点就是路径中间的那个节点
        # 否则则是中间的两个节点
        return [path[m // 2]] if m % 2 else [path[m // 2 - 1], path[m // 2]]

也可以使用dfs来找到最长路径

class Solution:
    def findMinHeightTrees(self, n: int, edges: List[List[int]]) -> List[int]:
        if n == 1: return [0]
        ref = [[] for _ in range(n)]
        for x, y in edges:
            ref[x].append(y)
            ref[y].append(x)
        parents = [0] * n
        self.maxDepth, self.node = 0, -1
        def dfs(start, parent, depth):
          	# 如果当前深度比最大深度大,则更新最大深度以及节点值
            if depth > self.maxDepth:
                self.maxDepth, self.node = depth, start
            parents[start] = parent
            for node in ref[start]:
                if node == parent: continue
                dfs(node, start, depth + 1)
        # 先找到0节点的最远距离,然后根据找到的节点再找最远距离的结点
        dfs(0, -1, 1)
        self.maxDepth = 0
        dfs(self.node, -1, 1)
        path = []
        node = self.node
        while node != -1:
            path.append(node)
            node = parents[node]
        m = len(path)
        return [path[m // 2]] if m % 2 else [path[m // 2 - 1], path[m // 2]]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

溪语流沙

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值