精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

二叉樹的所有路徑介紹

新材料在線 ? 來源:代碼隨想錄 ? 作者:程序員Carl ? 2021-08-13 17:51 ? 次閱讀

以為只用了遞歸,其實還用了回溯

257. 二叉樹的所有路徑

題目地址:https://leetcode-cn.com/problems/binary-tree-paths/

給定一個二叉樹,返回所有從根節點到葉子節點的路徑。

說明: 葉子節點是指沒有子節點的節點。

思路

這道題目要求從根節點到葉子的路徑,所以需要前序遍歷,這樣才方便讓父節點指向孩子節點,找到對應的路徑。

在這道題目中將第一次涉及到回溯,因為我們要把路徑記錄下來,需要回溯來回退一一個路徑在進入另一個路徑。

前序遍歷以及回溯的過程如圖:

07b5afe6-fbbe-11eb-9bcf-12bb97331649.png

我們先使用遞歸的方式,來做前序遍歷。要知道遞歸和回溯就是一家的,本題也需要回溯。

遞歸

遞歸函數函數參數以及返回值

要傳入根節點,記錄每一條路徑的path,和存放結果集的result,這里遞歸不需要返回值,代碼如下:

void traversal(TreeNode* cur, vector《int》& path, vector《string》& result)

確定遞歸終止條件

再寫遞歸的時候都習慣了這么寫:

if (cur == NULL) {

終止處理邏輯

}

但是本題的終止條件這樣寫會很麻煩,因為本題要找到葉子節點,就開始結束的處理邏輯了(把路徑放進result里)。

那么什么時候算是找到了葉子節點? 是當 cur不為空,其左右孩子都為空的時候,就找到葉子節點。

所以本題的終止條件是:

if (cur-》left == NULL && cur-》right == NULL) {

終止處理邏輯

}

為什么沒有判斷cur是否為空呢,因為下面的邏輯可以控制空節點不入循環。

再來看一下終止處理的邏輯。

這里使用vector結構path來記錄路徑,所以要把vector結構的path轉為string格式,在把這個string 放進 result里。

那么為什么使用了vector結構來記錄路徑呢? 因為在下面處理單層遞歸邏輯的時候,要做回溯,使用vector方便來做回溯。

可能有的同學問了,我看有些人的代碼也沒有回溯啊。

其實是有回溯的,只不過隱藏在函數調用時的參數賦值里,下文我還會提到。

這里我們先使用vector結構的path容器來記錄路徑,那么終止處理邏輯如下:

if (cur-》left == NULL && cur-》right == NULL) { // 遇到葉子節點

string sPath;

for (int i = 0; i 《 path.size() - 1; i++) { // 將path里記錄的路徑轉為string格式

sPath += to_string(path[i]);

sPath += “-》”;

}

sPath += to_string(path[path.size() - 1]); // 記錄最后一個節點(葉子節點)

result.push_back(sPath); // 收集一個路徑

return;

}

確定單層遞歸邏輯

因為是前序遍歷,需要先處理中間節點,中間節點就是我們要記錄路徑上的節點,先放進path中。

path.push_back(cur-》val);

然后是遞歸和回溯的過程,上面說過沒有判斷cur是否為空,那么在這里遞歸的時候,如果為空就不進行下一層遞歸了。

所以遞歸前要加上判斷語句,下面要遞歸的節點是否為空,如下

if (cur-》left) {

traversal(cur-》left, path, result);

}

if (cur-》right) {

traversal(cur-》right, path, result);

}

此時還沒完,遞歸完,要做回溯啊,因為path 不能一直加入節點,它還要刪節點,然后才能加入新的節點。

那么回溯要怎么回溯呢,一些同學會這么寫,如下:

if (cur-》left) {

traversal(cur-》left, path, result);

}

if (cur-》right) {

traversal(cur-》right, path, result);

}

path.pop_back();

這個回溯就要很大的問題,我們知道,回溯和遞歸是一一對應的,有一個遞歸,就要有一個回溯,這么寫的話相當于把遞歸和回溯拆開了, 一個在花括號里,一個在花括號外。

所以回溯要和遞歸永遠在一起,世界上最遙遠的距離是你在花括號里,而我在花括號外!

那么代碼應該這么寫:

if (cur-》left) {

traversal(cur-》left, path, result);

path.pop_back(); // 回溯

}

if (cur-》right) {

traversal(cur-》right, path, result);

path.pop_back(); // 回溯

}

那么本題整體代碼如下:

class Solution {private:

void traversal(TreeNode* cur, vector《int》& path, vector《string》& result) {

path.push_back(cur-》val);

// 這才到了葉子節點

if (cur-》left == NULL && cur-》right == NULL) {

string sPath;

for (int i = 0; i 《 path.size() - 1; i++) {

sPath += to_string(path[i]);

sPath += “-》”;

}

sPath += to_string(path[path.size() - 1]);

result.push_back(sPath);

return;

}

if (cur-》left) {

traversal(cur-》left, path, result);

path.pop_back(); // 回溯

}

if (cur-》right) {

traversal(cur-》right, path, result);

path.pop_back(); // 回溯

}

}

public

vector《string》 binaryTreePaths(TreeNode* root) {

vector《string》 result;

vector《int》 path;

if (root == NULL) return result;

traversal(root, path, result);

return result;

}

};

如上的C++代碼充分體現了回溯。

那么如上代碼可以精簡成如下代碼:

class Solution {private:

void traversal(TreeNode* cur, string path, vector《string》& result) {

path += to_string(cur-》val); // 中

if (cur-》left == NULL && cur-》right == NULL) {

result.push_back(path);

return;

}

if (cur-》left) traversal(cur-》left, path + “-》”, result); // 左

if (cur-》right) traversal(cur-》right, path + “-》”, result); // 右

}

public:

vector《string》 binaryTreePaths(TreeNode* root) {

vector《string》 result;

string path;

if (root == NULL) return result;

traversal(root, path, result);

return result;

}

};

如上代碼精簡了不少,也隱藏了不少東西。

注意在函數定義的時候void traversal(TreeNode* cur, string path, vector《string》& result) ,定義的是string path,每次都是復制賦值,不用使用引用,否則就無法做到回溯的效果。

那么在如上代碼中,貌似沒有看到回溯的邏輯,其實不然,回溯就隱藏在traversal(cur-》left, path + “-》”, result);中的 path + “-》”。 每次函數調用完,path依然是沒有加上“-》” 的,這就是回溯了。

為了把這份精簡代碼的回溯過程展現出來,大家可以試一試把:

if (cur-》left) traversal(cur-》left, path + “-》”, result); // 左 回溯就隱藏在這里

改成如下代碼:

path += “-》”;

traversal(cur-》left, path, result); // 左

即:

if (cur-》left) {

path += “-》”;

traversal(cur-》left, path, result); // 左

}

if (cur-》right) {

path += “-》”;

traversal(cur-》right, path, result); // 右

}

此時就沒有回溯了,這個代碼就是通過不了的了。

如果想把回溯加上,就要 在上面代碼的基礎上,加上回溯,就可以AC了。

if (cur-》left) {

path += “-》”;

traversal(cur-》left, path, result); // 左

path.pop_back(); // 回溯

path.pop_back();

}

if (cur-》right) {

path += “-》”;

traversal(cur-》right, path, result); // 右

path.pop_back(); // 回溯

path.pop_back();

}

大家應該可以感受出來,如果把 path + “-》”作為函數參數就是可以的,因為并有沒有改變path的數值,執行完遞歸函數之后,path依然是之前的數值(相當于回溯了)

綜合以上,第二種遞歸的代碼雖然精簡但把很多重要的點隱藏在了代碼細節里,第一種遞歸寫法雖然代碼多一些,但是把每一個邏輯處理都完整的展現了出來了。

迭代法

至于非遞歸的方式,我們可以依然可以使用前序遍歷的迭代方式來模擬遍歷路徑的過程,對該迭代方式不了解的同學,可以看文章二叉樹:聽說遞歸能做的,棧也能做!和二叉樹:前中后序迭代方式統一寫法。

這里除了模擬遞歸需要一個棧,同時還需要一個棧來存放對應的遍歷路徑。

C++代碼如下:

class Solution {public:

vector《string》 binaryTreePaths(TreeNode* root) {

stack《TreeNode*》 treeSt;// 保存樹的遍歷節點

stack《string》 pathSt; // 保存遍歷路徑的節點

vector《string》 result; // 保存最終路徑集合

if (root == NULL) return result;

treeSt.push(root);

pathSt.push(to_string(root-》val));

while (!treeSt.empty()) {

TreeNode* node = treeSt.top(); treeSt.pop(); // 取出節點 中

string path = pathSt.top();pathSt.pop(); // 取出該節點對應的路徑

if (node-》left == NULL && node-》right == NULL) { // 遇到葉子節點

result.push_back(path);

}

if (node-》right) { // 右

treeSt.push(node-》right);

pathSt.push(path + “-》” + to_string(node-》right-》val));

}

if (node-》left) { // 左

treeSt.push(node-》left);

pathSt.push(path + “-》” + to_string(node-》left-》val));

}

}

return result;

}

};

當然,使用java的同學,可以直接定義一個成員變量為object的棧Stack《Object》 stack = new Stack《》();,這樣就不用定義兩個棧了,都放到一個棧里就可以了。

總結

本文我們開始初步涉及到了回溯,很多同學過了這道題目,可能都不知道自己其實使用了回溯,回溯和遞歸都是相伴相生的。

我在第一版遞歸代碼中,把遞歸與回溯的細節都充分的展現了出來,大家可以自己感受一下。

第二版遞歸代碼對于初學者其實非常不友好,代碼看上去簡單,但是隱藏細節于無形。

最后我依然給出了迭代法。

對于本地充分了解遞歸與回溯的過程之后,有精力的同學可以在去實現迭代法。

其他語言版本

Java:

//解法一class Solution {

/**

* 遞歸法

*/

public List《String》 binaryTreePaths(TreeNode root) {

List《String》 res = new ArrayList《》();

if (root == null) {

return res;

}

List《Integer》 paths = new ArrayList《》();

traversal(root, paths, res);

return res;

}

private void traversal(TreeNode root, List《Integer》 paths, List《String》 res) {

paths.add(root.val);

// 葉子結點

if (root.left == null && root.right == null) {

// 輸出

StringBuilder sb = new StringBuilder();

for (int i = 0; i 《 paths.size() - 1; i++) {

sb.append(paths.get(i)).append(“-》”);

}

sb.append(paths.get(paths.size() - 1));

res.add(sb.toString());

return;

}

if (root.left != null) {

traversal(root.left, paths, res);

paths.remove(paths.size() - 1);// 回溯

}

if (root.right != null) {

traversal(root.right, paths, res);

paths.remove(paths.size() - 1);// 回溯

}

}

}

Python

class Solution:

def binaryTreePaths(self, root: TreeNode) -》 List[str]:

path=[]

res=[]

def backtrace(root, path):

if not root:return

path.append(root.val)

if (not root.left)and (not root.right):

res.append(path[:])

ways=[]

if root.left:ways.append(root.left)

if root.right:ways.append(root.right)

for way in ways:

backtrace(way,path)

path.pop()

backtrace(root,path)

return [“-》”.join(list(map(str,i))) for i in res]

Go:

func binaryTreePaths(root *TreeNode) []string {

res := make([]string, 0)

var travel func(node *TreeNode, s string)

travel = func(node *TreeNode, s string) {

if node.Left == nil && node.Right == nil {

v := s + strconv.Itoa(node.Val)

res = append(res, v)

return

}

s = s + strconv.Itoa(node.Val) + “-》”

if node.Left != nil {

travel(node.Left, s)

}

if node.Right != nil {

travel(node.Right, s)

}

}

travel(root, “”)

return res

}

責任編輯:haq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 函數
    +關注

    關注

    3

    文章

    4235

    瀏覽量

    61965
  • 二叉樹
    +關注

    關注

    0

    文章

    74

    瀏覽量

    12283

原文標題:二叉樹的所有路徑:不止遞歸,還有回溯

文章出處:【微信號:xincailiaozaixian,微信公眾號:新材料在線】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    指電極上覆蓋敏感材料的阻值計算

    覆蓋的敏感材料厚度超出指厚度時計算電阻,是否可以視作指電極指間電阻多個周期串聯后與超出指厚度部分敏感材料電阻并聯
    發表于 07-05 14:48

    指MOSFET器件靜電防護魯棒性提升技巧

    開啟,無法達到預期ESD防護等級。本文從版圖、器件結構、觸發技術等角度介紹一些改善多指MOSFET靜電防護器件電流泄放均勻性提升器件靜電防護魯棒性的技巧。
    的頭像 發表于 06-22 00:50 ?359次閱讀
    多<b class='flag-5'>叉</b>指MOSFET器件靜電防護魯棒性提升技巧

    哈夫曼編碼怎么算 哈夫曼編碼左邊是0還是1

    二叉樹,將出現頻率高的字符用較短的編碼表示,而出現頻率低的字符則用較長的編碼表示。通過這種方式,可以實現對數據進行高效的編碼和解碼。 下面我們將詳細介紹哈夫曼編碼的算法過程。 統計字符頻率 在進行哈夫曼編碼前,首先需
    的頭像 發表于 01-30 11:27 ?2068次閱讀

    時鐘是什么?介紹兩種時鐘樹結構

    今天來聊一聊時鐘。首先我先講一下我所理解的時鐘是什么,然后介紹兩種時鐘樹結構。
    的頭像 發表于 12-06 15:23 ?1336次閱讀

    weblogic修改jdk路徑

    路徑的情況。本文將詳細介紹如何在WebLogic中修改JDK路徑。 一、背景介紹 Java Development Kit(JDK)是Java開發人員必備的工具包,用于編譯、運行和調
    的頭像 發表于 12-05 14:46 ?1057次閱讀

    堆的實現思路

    什么是堆? 堆是一種 基于樹結構的數據結構,它是一棵二叉樹 ,具有以下兩個特點: 堆是一個完全二叉樹,即除了最后一層,其他層都是滿的,最后一層從左到右填滿。 堆中每個節點都滿足堆的特性,即父節點的值
    的頭像 發表于 11-24 16:02 ?331次閱讀
    堆的實現思路

    二叉樹的定義

    型結構 是一類重要的 非線性數據結構 ,其中以二叉樹最為常用,直觀來看,是以分支關系定義的層次結構。型結構在客觀世界中廣泛存在,比
    的頭像 發表于 11-24 15:57 ?1034次閱讀
    <b class='flag-5'>樹</b>與<b class='flag-5'>二叉樹</b>的定義

    OP-TEE安全存儲安全文件的格式

    時,默認情況下, 加密后的數據會被保存在/data/tee目錄中。 安全存儲功能使用 二叉樹的方式來 保存加密后的文件。 當第一次使用安全存儲功能創建用于保存敏感數據的安全文件時,OP-TEE將會在/data/tee目錄中生成兩個文件:dirf.db文件和以數字命名的文件。 dirf.db文
    的頭像 發表于 11-21 11:49 ?545次閱讀
    OP-TEE安全存儲安全文件的格式

    什么情況下需要布隆過濾器

    , gmail等郵箱垃圾郵件過濾功能 這幾個例子有一個共同的特點:如何判斷一個元素是否存在一個集合中? 常規思路 數組 鏈表 、平衡二叉樹、Trie Map (紅黑) 哈希表 雖然上面描述的這幾種數據結構配合常見的排序、
    的頭像 發表于 11-11 11:37 ?552次閱讀
    什么情況下需要布隆過濾器

    紅黑的特點及應用

    比起理解紅黑的原理,更重要的是理解紅黑的應用場景,因為某些應用場景的需要,紅黑才會應運而生。 紅黑的特點: 插入,刪除,查找都是O(logn)的復雜度。 紅黑
    的頭像 發表于 11-10 11:16 ?609次閱讀
    紅黑<b class='flag-5'>樹</b>的特點及應用

    為什么MySQL索引要用B+tree?

    紅黑是一種特化的 AVL(平衡二叉樹),都是在進行插入和刪除操作時通過特定操作保持二叉查找的平衡; 若一棵
    發表于 10-30 14:41 ?170次閱讀

    路徑和iSCSI SAN存儲技術介紹

    driver和設備文件著手,告訴了操作系統怎么來處理這些身份復雜的LUN。 上篇“存儲基礎和FC SAN存儲介紹”重點介紹FC存儲技術。今天,簡單給小伙伴普及下Linux系統Multipath多路徑軟件和多
    的頭像 發表于 10-24 11:08 ?919次閱讀
    多<b class='flag-5'>路徑</b>和iSCSI SAN存儲技術<b class='flag-5'>介紹</b>

    文件系統-多二叉樹的轉化

    在這一節中,我們來學習如何使用程序來實現一棵文件。在上一節中,我們了解到使用文件的方式來整合計算機中所有的資源,而這一棵文件則是一棵多
    的頭像 發表于 10-11 10:06 ?768次閱讀
    文件系統-多<b class='flag-5'>叉</b><b class='flag-5'>樹</b>與<b class='flag-5'>二叉樹</b>的轉化

    數據結構面試之二叉樹相關操作

    根據前序可知根結點為1; 根據中序可知 4 7 2 為根結點 1 的左子樹和 8 5 9 3 6 為根結點 1 的右子樹; 遞歸實現,把 4 7 2 當做新的一棵和 8 5 9 3 6 也當做新的一棵; 在遞歸的過程中輸出后序。
    發表于 10-10 14:50 ?182次閱讀
    數據結構面試之<b class='flag-5'>二叉樹</b>相關操作

    物聯網開發需要學習哪些內容?

    和需要掌握的技能。 1. 物聯網軟件開發必備編程技術: Linux C語言、數據結構 核心技能內容: 必備的Linux命令; C語言的基礎知識; C語言的數組、指針和函數; 數據結構中的線性表、棧和隊列用法及實現; 二叉樹遞歸遍歷、層次遍歷、及非
    的頭像 發表于 10-09 17:23 ?1388次閱讀