剑指offer - 二叉树中和为某一值的路径 - JavaScript

二叉树中和为某一值的路径

http://www.nowcoder.com/questionTerminal/b736e784e3e34731af99065031301bca

【二叉树中和为某一值的路径】【2种解法:递归 + 非递归】

解法 1: 前序遍历(递归)

算法实现思路是:

  • 每次来到新节点,将节点放入当前保存的路径
  • 检查节点是否是叶节点:
    • 是:将路径放入结果中
    • 不是:继续遍历左子树和右子树

上面整个过程就是一个前序遍历,但在遍历的过程中,动态地维护了当前路径与总和的信息。

在实现过程中需要注意的是,JavaScript 中传入数组做参数,函数内拿到的是数组的引用,不是深拷贝。代码实现如下:

// ac地址:https://www.nowcoder.com/practice/b736e784e3e34731af99065031301bca
// 原文地址:https://xxoo521.com/2020-02-04-btree-find-path/

/* function TreeNode(x) {
    this.val = x;
    this.left = null;
    this.right = null;
} */

function FindPath(root, expectNumber) {
    if (!root) {
        return [];
    }
    const pathes = [];
    __FindPath(root, expectNumber, pathes, []);
    return pathes;
}

function __FindPath(root, expectNumber, pathes, path) {
    if (!root) {
        return;
    }

    path = [...path, root.val]; // 深拷贝

    if (!root.left && !root.right && root.val === expectNumber) {
        pathes.push(path);
        return;
    }

    __FindPath(root.left, expectNumber - root.val, pathes, path);
    __FindPath(root.right, expectNumber - root.val, pathes, path);
}

解法 2: 非递归遍历

这里可以借助栈,改造成非递归前序遍历。从而减少运行时间。

为了方便表示节点信息,栈中的元素是形如(node, 剩余路径和, 节点路径组成的数组)这样的元祖。

整体的处理流程是:

  • 取出栈顶的元祖:节点、剩余路径和、路径
  • 如果当前节点是叶节点,且剩余路径和等于节点的 val,那么将路径放入结果中
  • 如果右节点不为空,将(右节点,剩余路径和 - 右节点值,路径+右节点)放入栈中
  • 如果做节点不为空,处理过程和右节点一样

注意,为什么先处理右节点而不是左节点?先遍历左和右都可以。但是因为用的牛客网的 oj 平台,这里要求“数组长度大的数组靠前”,先遍历左节点就是 oc 不了。

代码实现如下:

// ac地址:https://www.nowcoder.com/practice/b736e784e3e34731af99065031301bca
// 原文地址:https://xxoo521.com/2020-02-04-btree-find-path/

/* function TreeNode(x) {
    this.val = x;
    this.left = null;
    this.right = null;
} */

function FindPath(root, expectNumber) {
    if (!root) {
        return [];
    }

    const statck = [[root, expectNumber, [root.val]]];
    const result = [];

    while (statck.length) {
        const [node, num, path] = statck.pop();

        if (!node.left && !node.right && node.val === num) {
            result.push(path);
        }

        if (node.right) {
            statck.push([
                node.right,
                num - node.val,
                [...path, node.right.val]
            ]);
        }
        if (node.left) {
            statck.push([node.left, num - node.val, [...path, node.left.val]]);
        }
    }

    return result;
}

拓展思考:回溯剪枝

如果题目说明了:所有节点的值都是正数,那么在递归遍历的时候,就可以进行回溯剪枝。

剪枝的方法就是如果发现当前的节点的值大于 expectNumber,停止递归。

改动后的__FindPath()的代码如下:

// 原文地址:https://xxoo521.com/2020-02-04-btree-find-path/
function __FindPath(root, expectNumber, pathes, path) {
    if (!root || root.val > expectNumber) {
        // 剪枝
        return;
    }

    path = [...path, root.val];

    if (!root.left && !root.right && root.val === expectNumber) {
        pathes.push(path);
        return;
    }

    __FindPath(root.left, expectNumber - root.val, pathes, path);
    __FindPath(root.right, expectNumber - root.val, pathes, path);
}

有意思的是这段代码在牛客网平台可以 oc,测试用例不全面。

博客地址:剑指 Offer 题解+代码

收藏 Github :https://github.com/dongyuanxin/blog

全部评论
...是浅拷贝吧大哥
点赞 回复 分享
发布于 2020-07-28 21:19

相关推荐

点赞 评论 收藏
分享
牛客773130651号:巨佬,简历模板换成上下的,左右的很烦,hr看着不爽。。。科大随便乱杀,建议能保研就保研,不行也得考一下 ,985硕去干算法,比开发强多了。开发许多双非都能搞,学历优势用不上,算法有门槛
点赞 评论 收藏
分享
缓解焦虑的最好方法是回家。鼠鼠昨天上午考完了本科阶段的最后一场考试,大概率考得稀烂,但是没多想,考完立马收拾行李,坐上了提前约好的顺风车飞奔回家。虽然家和学校很近,只有一百多公里的路程,但距离上次回家也已经有三四个月了。每次想回家,期间总有考试、毕业设计、面试、实习等等各种各样的原因,没办法回去,待在学校和公司的每一天也都充斥着无形的压力和焦虑。现在终于完成了答辩,考完了试,公司那边也请了假,是时候回去一趟了。没有提前通知爸妈,想给他们一个惊喜。下午提前到了家,他俩还在上班,只好让外公外婆来给我开门。因为我的回家,晚上外婆在厨房格外忙碌,做了满满一大桌子菜,填饱了我天天吃外卖的肚子。晚上也没空...
梦想是成为七海千秋:取决于家庭吧?其实回家更焦虑了,每天起床父母都问实习找好了没简历投递了没今天有没有面试,但是又没有什么结果,玩两下手机父母就会说你看你啥也没找到为什么天天就知道刷手机,怎么不去学习…我现在就希望我能永远在外面实习,报喜不报忧,等拿到一个好offer再回家
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务