979 在二叉树中分配硬币-中等

题目:

给你一个有 n 个结点的二叉树的根结点 root ,其中树中每个结点 node 都对应有 node.val 枚硬币。整棵树上一共有 n 枚硬币。

在一次移动中,我们可以选择两个相邻的结点,然后将一枚硬币从其中一个结点移动到另一个结点。移动可以是从父结点到子结点,或者从子结点移动到父结点。

返回使每个结点上 只有 一枚硬币所需的 最少 移动次数。

分析:

从流通的角度计算。

// date 2023/10/28
/**
 * Definition for a binary tree node.
 * type TreeNode struct {
 *     Val int
 *     Left *TreeNode
 *     Right *TreeNode
 * }
 */
func distributeCoins(root *TreeNode) int {
    var ans int

    // 以每个节点目标为1开始计算,
    // 计算该节点所在的子树每个节点都满足1,所需要或者贡献出去的金币数
    var dfs func(root *TreeNode) int
    dfs = func(root *TreeNode) int {
        if root == nil {
            return 0
        }
        // 每个节点最终都要有 1 个
        // 1 表示需要 1 个, 0 表示不需要, 负数表示需要贡献出去的
        need := 1 - root.Val
        l := dfs(root.Left)  // left  贡献或者需要
        r := dfs(root.Right) // right 贡献或者需要的

        // 在当前节点的视角看来,左右之间的不平衡就是需要 流桶 的次数
        ans += abs(l) + abs(r)
        
        // 跳出当前节点的视角,从外部看该节点
        // 左右相加,加上节点本身,就是该节点整体对外需要或者贡献的
        return l + r + need
    }

    dfs(root)

    return ans
}

func abs(x int) int {
    if x > 0 {
        return x
    }
    return -x
}

最后更新于