leetcode NO.124 二叉树中的最大路径和 白痴讲解 腾讯精选练习50

在这里插入图片描述
这是官方解答的代码
https://leetcode-cn.com/problems/binary-tree-maximum-path-sum/solution/er-cha-shu-de-zui-da-lu-jing-he-by-leetcode/

# Definition for a binary tree node.
# class TreeNode:
#     def __init__(self, x):
#         self.val = x
#         self.left = None
#         self.right = None

class Solution:
    def maxPathSum(self, root):
        """
        :type root: TreeNode
        :rtype: int
        """
        def max_gain(node):
            nonlocal max_sum
            #nonlocal的作用是,当函数内嵌套函数时,用nonlocal声明 max_sum 是局部变量,会根据上一次函数的更新而更新
            if not node:
                return 0

            # max sum on the left and right sub-trees of node
            left_gain = max(max_gain(node.left), 0)
            right_gain = max(max_gain(node.right), 0)
            
            # the price to start a new path where `node` is a highest node
            price_newpath = node.val + left_gain + right_gain
            
            # update max_sum if it's better to start a new path
            max_sum = max(max_sum, price_newpath)
        
            # for recursion :
            # return the max gain if continue the same path
            return node.val + max(left_gain, right_gain)
   
        max_sum = float('-inf')
        max_gain(root)
        return max_sum

max_sum是我们要的答案,我们先初始化为一个最小的float,方便我们替换,因为我们找的是最大的路径,用最小的float来初始化,无论碰到什么数我们都会更新它。
我们来讲一下怎么递归
首先max_gain(node)这个函数的作用是以node为根节点,也就是树的最上端那个点形成的这棵二叉树,它是取来自左侧子树从下往上的最大路径还是取来自右侧子树从下往上的最大路径,加这个node点本身,可以获得的最大路径值,并返回这个最大路径值。
解释一下这里的从下往上,就是这条路径只有从子叶节点到父叶节点再到现在这个根节点node这种,不包含可以回头的那种,就是子叶节点到父叶节点又回去子叶节点的那种情况。
举例说明就是
以-10这个node为例
在这里插入图片描述
max_gain(-10为根节点的这棵树)是得到来自左侧子树从下往上的最大路径(9)和来自右侧子树从下往上的最大路径(15+20),不包含(15+20+7)这种回头的情况。(15+20+7)这种情况我们在递归中会介绍怎么处理。
根据我们之前给出的数学归纳法解法
我们要给出n = 0 的时候的解,对应碰到最后一层叶子节点,再调用函数就到None了,所以return 0

if not node:
    return 0

n = k 时 如何推到 n = k + 1 呢

left_gain = max(max_gain(node.left), 0)
right_gain = max(max_gain(node.right), 0)
price_newpath = node.val + left_gain + right_gain

从这个node开始,我们能从左侧子树得到什么呢?
如果来自左侧子树的从下往上的最大路径比0还要小,右侧子树可以带来正数的路径和,还完全没有必要选左边边的子树了,因为加起来必定比node本身要小,我们之选右边
同理我们可以处理来自右侧子树的从下网上的最大路径
还有可能我只选这个node,左右两边都是负收益
那一开始(15+20+7)那种情况怎么处理呢?这种就对应着left_gain和right_gain都是有利可图的,也就是我放弃继续往上走了,我回去找左侧到节点node到右侧的这种路径
对于递归到最底下的叶子结点,price_newpath就是叶子结点本身的值
上面的各种情况都能用这句话来计算最大路径和,肥肠牛逼!
我们用一个新路径的价格的变量来记录它,如果它比历史的最大路径和大,那么就改变max_sum的值为这个值

max_sum = max(max_sum, price_newpath)

但是我们的函数最后还是要返回n = k + 1时候的max_gain()
等于node加上来是选左侧子树还是右侧子树从下往上的最大路径和

return node.val + max(left_gain, right_gain)

实际上递归完成,同时也更新完了最大路径和。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值