LeetCode - 94. 二叉樹的中序遍歷

題目

94. 二叉樹的中序遍歷 - 力扣(LeetCode)

什么是中序遍歷

二叉樹的中序遍歷是按照"左-根-右"的順序訪問二叉樹中的所有節點。

具體過程:

  • 先遍歷左子樹(遞歸)
  • 然后訪問根節點
  • 最后遍歷右子樹(遞歸)

例如,對于下面的二叉樹:

    1/ \2   3/ \   \
4   5   6

中序遍歷的結果是:[4, 2,?5, 1, 3,?6]

遍歷步驟:

  • 從根節點開始,先訪問左子樹
  • 訪問節點4(左葉子)
  • 訪問節點2(4的父節點)
  • 訪問節點5(2的右子節點)
  • 訪問節點1(根節點)
  • 訪問節點3(右子節點)
  • 訪問節點6(3的右子節點)

遞歸寫法

讀者可能會出現的錯誤寫法

class Solution {
public:vector<int> inorderTraversal(TreeNode* root) {vector<int> result; dfs(root,result);return result;}void dfs(TreeNode* root,vector<int> result){if(!root){return;}dfs(root->left,result);result.push_back(root->val);dfs(root->right,result);}
};

在dfs函數中,result參數是按值傳遞的,而不是按引用傳遞的,這會導致對result的修改不會保存。

每次調用dfs函數時:

  1. 會創建result的一個新副本
  2. 當執行result.push_back(root->val)時,只是向這個副本中添加元素
  3. 遞歸調用子節點的dfs時,又會創建新的副本
  4. 當函數返回時,這個副本被銷毀,其中存儲的所有節點值都消失了

正確寫法

class Solution {
public:vector<int> inorderTraversal(TreeNode* root) {vector<int> result; dfs(root,result);return result;}void dfs(TreeNode* root,vector<int>& result){if(!root){return;}dfs(root->left,result);result.push_back(root->val);dfs(root->right,result);}
};

非遞歸寫法

思路

  • 沿著左子樹一直深入,將所有節點壓入棧中
  • 當無法繼續向左時,彈出棧頂節點,訪問它
  • 然后處理該節點的右子樹

具體實現步驟:

  • 創建一個空棧和一個指向當前節點的指針curr(初始為根節點)
  • 當curr不為空或棧不為空時循環:
  • 如果curr不為空,將curr壓入棧,然后curr移動到其左子節點
  • 如果curr為空,彈出棧頂節點,將其值加入結果數組,然后curr移動到其右子節點

讀者可能的錯誤寫法

class Solution {
public:vector<int> inorderTraversal(TreeNode* root) {vector<int> result;stack<TreeNode*>st;TreeNode* node = root;while(node || !st.empty()){while(node){st.push(node);node=node->left;}//將棧頂的元素放入數組中result.push_back(node->val);st.pop();node = node->right;}return result;}
};

在嘗試訪問node->val之前,node已經是nullptr。在內層while循環結束后,node已經是nullptr,因為循環的終止條件就是node為空。因此,在這之后直接訪問node->val和node->right會導致空指針解引用錯誤。

正確的方法

vector<int> inorderTraversal(TreeNode* root) 
{vector<int> result;stack<TreeNode*> st;TreeNode* node = root;while(node || !st.empty()){while(node){st.push(node);node = node->left;}node = st.top();  // 先獲取棧頂節點st.pop();result.push_back(node->val);node = node->right;}return result;
}

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/908738.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/908738.shtml
英文地址,請注明出處:http://en.pswp.cn/news/908738.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

PyTorch——搭建小實戰和Sequential的使用(7)

import torch from torch import nn from torch.nn import Conv2d, MaxPool2d, Flatten, Linearclass TY(nn.Module):def __init__(self):"""初始化TY卷積神經網絡模型模型結構&#xff1a;3層卷積池化&#xff0c;2層全連接設計目標&#xff1a;處理32x32像素的…

C#、VB.net——如何設置窗體應用程序的外邊框不可拉伸

以Visual studio 2015為例&#xff0c;具體操作如下&#xff1a; 1、將窗體的“FormBorderStyle”屬性值修改為“FixedSingle”&#xff1a; 2、點擊“格式”——“鎖定控件”&#xff1a; 這樣生成的程序邊框即可固定住&#xff0c;無法拉伸。

深入了解NIO的優化實現原理

網絡 I/O 模型優化 網絡通信中&#xff0c;最底層的就是內核中的網絡 I/O 模型了。隨著技術的發展&#xff0c;操作系統內核的網絡模型衍生出了五種 I/O 模型&#xff0c;《UNIX 網絡編程》一書將這五種 I/O 模型分為阻塞式 I/O、非阻塞式 I/O、I/O 復用、信號驅動式 I/O 和異步…

【前端】vue3性能優化方案

以下是Vue 3性能優化的系統性方案&#xff0c;結合核心優化策略與實用技巧&#xff0c;覆蓋渲染、響應式、加載、代碼等多個維度&#xff1a; ?? 一、渲染優化 精準控制渲染范圍 v-if vs v-show&#xff1a; v-if&#xff1a;條件為假時銷毀DOM&#xff0c;適合低頻切換場景&…

在MATLAB中使用自定義的ROS2消息

簡明結論&#xff1a; 無論ROS2節點和MATLAB運行在哪&#xff0c;MATLAB本機都必須擁有自定義消息源碼并本地用ros2genmsg生成&#xff0c;才能在Simulink里訂閱這些消息。只要你想讓MATLAB或Simulink能識別自定義消息&#xff0c;必須把消息包源碼(.msg等)拷到本機指定目錄&a…

spring重試機制

數據庫死鎖處理與重試機制實現指南 1. 業務場景 1.1 問題現象 高并發批量數據處理時頻繁出現數據庫死鎖主要發生在"先刪除歷史數據&#xff0c;再重新計算"的業務流程中原有逐條處理方式&#xff1a;list.forEach(item -> { delete(); calculate(); }) 1.2 死…

QEMU源碼全解析 —— 塊設備虛擬化(24)

接前一篇文章:QEMU源碼全解析 —— 塊設備虛擬化(23) 本文內容參考: 《趣談Linux操作系統》 —— 劉超,極客時間 《QEMU/KVM源碼解析與應用》 —— 李強,機械工業出版社 特此致謝! QEMU寫入一個文件的完整過程 前邊用了十來篇文章的篇幅,解析了QEMU啟動過程中的存儲…

java中static學習筆記

較重要知識點 static修飾的變量是共享的在類加載時創建可以不通過實例來訪問靜態方法只能訪問靜態的成員和方法&#xff1b;而非靜態的可以訪問靜態的和非靜態的。靜態方法一般用在通用的方法&#xff0c;這樣方便調用&#xff0c;不然一個通用的方法每一次調用都要創建實例&a…

快刀集(1): 一刀斬斷視頻片頭廣告

一刀流&#xff1a;用一個簡單腳本&#xff0c;秒殺視頻片頭廣告&#xff0c;還你清爽觀影體驗。 1. 引子 作為一個愛生活、愛學習、愛收藏高清資源的老碼農&#xff0c;平時寫代碼之余看看電影、補補片&#xff0c;是再正常不過的事。 電影嘛&#xff0c;要沉浸&#xff0c;…

spring中的@KafkaListener 注解詳解

KafkaListener 是 Spring Kafka 提供的一個核心注解&#xff0c;用于標記一個方法作為 Kafka 消息的消費者。下面是對該注解的詳細解析&#xff1a; 基本用法 KafkaListener(topics "myTopic", groupId "myGroup") public void listen(String message)…

多區域協同的異地多活AI推理服務架構

&#x1f310;多區域協同的異地多活AI推理服務架構 #mermaid-svg-TTnpRKKC7k3twxhE {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-TTnpRKKC7k3twxhE .error-icon{fill:#552222;}#mermaid-svg-TTnpRKKC7k3twxhE .er…

極客時間:在 Google Colab 上嘗試 Prefix Tuning

每周跟蹤AI熱點新聞動向和震撼發展 想要探索生成式人工智能的前沿進展嗎&#xff1f;訂閱我們的簡報&#xff0c;深入解析最新的技術突破、實際應用案例和未來的趨勢。與全球數同行一同&#xff0c;從行業內部的深度分析和實用指南中受益。不要錯過這個機會&#xff0c;成為AI領…

Android設備推送traceroute命令進行網絡診斷

文章目錄 工作原理下載traceroute for android推送到安卓設備執行traceroutetraceroute www.baidu.com Traceroute&#xff08;追蹤路由&#xff09; 是一個用于網絡診斷的工具&#xff0c;主要用于追蹤數據包從源主機到目標主機所經過的路由路徑&#xff0c;以及每一跳&#x…

【Linux應用】Linux系統日志上報服務,以及thttpd的配置、發送函數

【Linux應用】Linux系統日志上報服務&#xff0c;以及thttpd的配置、發送函數 文章目錄 thttpd服務安裝thttpd配置thttpd服務thttpd函數日志效果和文件附錄&#xff1a;開發板快速上手&#xff1a;鏡像燒錄、串口shell、外設掛載、WiFi配置、SSH連接、文件交互&#xff08;RADX…

Linux 內核內存管理子系統全面解析與體系構建

一、前言: 為什么內存管理是核心知識 內存管理是 Linux 內核最核心也最復雜的子系統之一&#xff0c;其作用包括&#xff1a; 為軟件提供獨立的虛擬內存空間&#xff0c;實現安全隔離分配/回收物理內存資源&#xff0c;維持系統穩定支持不同類型的內存分配器&#xff0c;最優…

鼠標的拖動效果

1、變量的設置 let isDragging false; let startX; let startY&#xff1b; let endX; let endY; let box null;isDragging : 表示是否推拽startX、startY&#xff1a;表示起始坐標&#xff0c;相對于元素endX、endY&#xff1a;表示結束坐標&#xff0c;相對于元素box&…

SwaggerFuzzer:一款自動化 OpenAPI/Swagger 接口未授權訪問測試工具

SwaggerFuzzer &#x1f310; 一款自動化 OpenAPI/Swagger 接口未授權訪問測試工具&#x1f680; 工具介紹&#xff1a;SwaggerFuzzer? 核心功能亮點&#x1f680; 快速使用&#x1f9f0; 支持參數 &#x1f4cc; 項目結構&#x1f4e5; 獲取與下載 &#x1f310; 一款自動化 …

文獻閱讀:Exploring Autoencoder-based Error-bounded Compression for Scientific Data

目錄 論文簡介動機&#xff1a;為什么作者想要解決這個問題&#xff1f;貢獻&#xff1a;作者在這篇論文中完成了什么工作(創新點)&#xff1f;規劃&#xff1a;他們如何完成工作&#xff1f;離線訓練階段&#xff1a;在線壓縮階段 理由&#xff1a;通過什么實驗驗證它們的工作…

【業務框架】3C-相機-Cinemachine

概述 插件&#xff0c;做相機需求&#xff0c;等于相機老師傅多年經驗總結的工具 Feature Transform&#xff1a;略Control Camera&#xff1a;控制相機參數Noise&#xff1a;增加隨機性Blend&#xff1a;CameraBrain的混合列表指定一個虛擬相機到另一個相機的過渡&#xff…

設計一個算法:刪除非空單鏈表L中結點值為x的第一個結點的前驅結點

目錄 單鏈表的存儲結構定義如下 快慢指針法 三指針法版本① 三指針法版本② 單鏈表的存儲結構定義如下 typedef struct{Elemtype data;struct Node* next; }LNode,*LinkList; 快慢指針法 void deleteprex(LinkList L, Elemtype e) {if (L NULL || L->next NULL ||…