樹的結點查找和刪除

来源:https://www.cnblogs.com/malinyan/archive/2022/09/25/BinaryTreeSearchAndDelete.html
-Advertisement-
Play Games

二叉樹查找指定的節點 前序查找的思路 1.先判斷當前節點的no是否等於要查找的 2.如果是相等,則返回當前節點 3.如果不等,則判斷當前節點的左子節點是否為空,如果不為空,則遞歸前序查找 4.如果左遞歸前序查找,找到節點,則返回,否繼續判斷,當前的節點的右子節點是否為空,如果不為空,則繼續向右遞歸前 ...


二叉樹查找指定的節點

前序查找的思路

1.先判斷當前節點的no是否等於要查找的

2.如果是相等,則返回當前節點

3.如果不等,則判斷當前節點的左子節點是否為空,如果不為空,則遞歸前序查找

4.如果左遞歸前序查找,找到節點,則返回,否繼續判斷,當前的節點的右子節點是否為空,如果不為空,則繼續向右遞歸前序查找。

中序查找思路

1.判斷當前節點的左子節點是否為空,如果不為空,則遞歸中序查找

2.如果找到,則返回,如果沒有找到,就和當前節點比較,如果是則返回當前節點,否則繼續進行右遞歸的中序查找

3.如果右遞歸中序查找,找到就返回,否則返回null

後序查找思路

1.判斷當前節點的左子節點是否為空,如果不為空,則遞歸後序查找

2.如果找到,就返回,如果沒有找到,就判斷當前節點的右子節點是否為空,如果不為空,則右遞歸進行後序查找,如果找到,就返回

3.就和當前節點進行比較,如果是則返回,否則返回null

要求

1.請編寫前序查找,中序查找和後序查找的方法。

2.並分別使用三種查找方式,查找 heroNO = 5 的節點

3.並分析各種查找方式,分別比較了多少次

代碼實現:

先創建HeroNode 結點

class HeroNode {
	private int no;
	private String name;
	private HeroNode left; //預設null
	private HeroNode right; //預設null
	public HeroNode(int no, String name) {
		this.no = no;
		this.name = name;
	}
	public int getNo() {
		return no;
	}
	public void setNo(int no) {
		this.no = no;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public HeroNode getLeft() {
		return left;
	}
	public void setLeft(HeroNode left) {
		this.left = left;
	}
	public HeroNode getRight() {
		return right;
	}
	public void setRight(HeroNode right) {
		this.right = right;
	}
	@Override
	public String toString() {
		return "HeroNode [no=" + no + ", name=" + name + "]";
	}
    //前序遍歷
	public void preOrder() {
		System.out.println(this); //先輸出父結點
		//遞歸向左子樹前序遍歷
		if(this.left != null) {
			this.left.preOrder();
		}
		//遞歸向右子樹前序遍歷
		if(this.right != null) {
			this.right.preOrder();
		}
	}
	//前序遍歷查找
	/**
	 * 
	 * @param no 查找no
	 * @return 如果找到就返回該Node ,如果沒有找到返回 null
	 */
	public HeroNode preOrderSearch(int no) {
		System.out.println("進入前序遍歷");
		//比較當前結點是不是
		if(this.no == no) {
			return this;
		}
		//1.則判斷當前結點的左子節點是否為空,如果不為空,則遞歸前序查找
		//2.如果左遞歸前序查找,找到結點,則返回
		HeroNode resNode = null;
		if(this.left != null) {
			resNode = this.left.preOrderSearch(no);
		}
		if(resNode != null) {//說明我們左子樹找到
			return resNode;
		}
		//1.左遞歸前序查找,找到結點,則返回,否繼續判斷,
		//2.當前的結點的右子節點是否為空,如果不空,則繼續向右遞歸前序查找
		if(this.right != null) {
			resNode = this.right.preOrderSearch(no);
		}
		return resNode;
	}
	
	//中序遍歷查找
	public HeroNode infixOrderSearch(int no) {
		//判斷當前結點的左子節點是否為空,如果不為空,則遞歸中序查找
		HeroNode resNode = null;
		if(this.left != null) {
			resNode = this.left.infixOrderSearch(no);
		}
		if(resNode != null) {
			return resNode;
		}
		System.out.println("進入中序查找");
		//如果找到,則返回,如果沒有找到,就和當前結點比較,如果是則返回當前結點
		if(this.no == no) {
			return this;
		}
		//否則繼續進行右遞歸的中序查找
		if(this.right != null) {
			resNode = this.right.infixOrderSearch(no);
		}
		return resNode;
		
	}
	
	//後序遍歷查找
	public HeroNode postOrderSearch(int no) {
		
		//判斷當前結點的左子節點是否為空,如果不為空,則遞歸後序查找
		HeroNode resNode = null;
		if(this.left != null) {
			resNode = this.left.postOrderSearch(no);
		}
		if(resNode != null) {//說明在左子樹找到
			return resNode;
		}
		
		//如果左子樹沒有找到,則向右子樹遞歸進行後序遍歷查找
		if(this.right != null) {
			resNode = this.right.postOrderSearch(no);
		}
		if(resNode != null) {
			return resNode;
		}
		System.out.println("進入後序查找");
		//如果左右子樹都沒有找到,就比較當前結點是不是
		if(this.no == no) {
			return this;
		}
		return resNode;
	}
	
}

定義BinaryTree 二叉樹

class BinaryTree {
	private HeroNode root;

	public void setRoot(HeroNode root) {
		this.root = root;
	}
	
	//前序遍歷查找
	public HeroNode preOrderSearch(int no) {
		if(root != null) {
			return root.preOrderSearch(no);
		} else {
			return null;
		}
	}
	//中序遍歷查找
	public HeroNode infixOrderSearch(int no) {
		if(root != null) {
			return root.infixOrderSearch(no);
		}else {
			return null;
		}
	}
	//後序遍歷查找
	public HeroNode postOrderSearch(int no) {
		if(root != null) {
			return this.root.postOrderSearch(no);
		}else {
			return null;
		}
	}
}

測試:

public class BinaryTreeDemo {

	public static void main(String[] args) {
		//先需要創建一顆二叉樹
		BinaryTree binaryTree = new BinaryTree();
		//創建需要的結點
		HeroNode root = new HeroNode(1, "宋江");
		HeroNode node2 = new HeroNode(2, "吳用");
		HeroNode node3 = new HeroNode(3, "盧俊義");
		HeroNode node4 = new HeroNode(4, "林沖");
		HeroNode node5 = new HeroNode(5, "關勝");
		
		//說明,我們先手動創建該二叉樹,後面我們學習遞歸的方式創建二叉樹
		root.setLeft(node2);
		root.setRight(node3);
		node3.setRight(node4);
		node3.setLeft(node5);
		binaryTree.setRoot(root);
		
		// 前序遍歷
		// 前序遍歷的次數 :4
		System.out.println("前序遍歷方式~~~");
		HeroNode resNode = binaryTree.preOrderSearch(5);
		if (resNode != null) {
			System.out.printf("找到了,信息為 no=%d name=%s", resNode.getNo(), resNode.getName());
		} else {
			System.out.printf("沒有找到 no = %d 的英雄", 5);
		}
		
		/**
			// 中序遍歷查找
			// 中序遍歷3次
			System.out.println("中序遍歷方式~~~");
			HeroNode resNode = binaryTree.infixOrderSearch(5);
			if (resNode != null) {
				System.out.printf("找到了,信息為 no=%d name=%s", resNode.getNo(), resNode.getName());
			} else {
				System.out.printf("沒有找到 no = %d 的英雄", 5);
			}
		*/
		/**
			// 後序遍歷查找
			// 後序遍歷查找的次數 2次
			System.out.println("後序遍歷方式~~~");
			HeroNode resNode = binaryTree.postOrderSearch(5);
			if (resNode != null) {
				System.out.printf("找到了,信息為 no=%d name=%s", resNode.getNo(), resNode.getName());
			} else {
				System.out.printf("沒有找到 no = %d 的英雄", 5);
			}
		*/
		
	}

}

運行結果如圖:

前序遍歷:

中序遍歷:

後序遍歷:

二叉樹刪除指定的節點

要求

1.如果刪除的節點是葉子節點,則刪除該節點

2.如果刪除的節點是非葉子節點,則刪除該子樹.

3.測試,刪除掉 5 號葉子節點

思路:

1.考慮如果樹是空樹root,如果只有一個root節點,則等價將二叉樹置空

2.因為我們的二叉樹是單向的,所以我們是判斷當前節點的子節點是否是需要刪除節點,而不能去判斷當前這個節點是不是需要刪除節點。

3.如果當前節點的左子節點不為空,並且左子節點就是要刪除節點,就將this.left=null;

並且就返回(結束遞歸刪除)

4.如果當前節點的右子節點不為空,並且右子節點就是要刪除節點,就將this.right=null;

並且就返回(結束遞歸刪除)

5.如果3,4步沒有刪除節點,那麼我們就需要向左子樹進行遞歸刪除

6.如果第5步也沒有刪除節點,則應當向右子樹進行遞歸刪除。

代碼實現:

//HeroNode 類增加方法

//遞歸刪除結點
	//1.如果刪除的節點是葉子節點,則刪除該節點
	//2.如果刪除的節點是非葉子節點,則刪除該子樹
	public void delNode(int no) {
		
		//思路
		/*
		 * 	1. 因為我們的二叉樹是單向的,所以我們是判斷當前結點的子結點是否需要刪除結點,而不能去判斷當前這個結點是不是需要刪除結點.
			2. 如果當前結點的左子結點不為空,並且左子結點 就是要刪除結點,就將this.left = null; 並且就返回(結束遞歸刪除)
			3. 如果當前結點的右子結點不為空,並且右子結點 就是要刪除結點,就將this.right= null ;並且就返回(結束遞歸刪除)
			4. 如果第2和第3步沒有刪除結點,那麼我們就需要向左子樹進行遞歸刪除
			5.  如果第4步也沒有刪除結點,則應當向右子樹進行遞歸刪除.

		 */
		//2. 如果當前結點的左子結點不為空,並且左子結點 就是要刪除結點,就將this.left = null; 並且就返回(結束遞歸刪除)
		if(this.left != null && this.left.no == no) {
			this.left = null;
			return;
		}
		//3.如果當前結點的右子結點不為空,並且右子結點 就是要刪除結點,就將this.right= null ;並且就返回(結束遞歸刪除)
		if(this.right != null && this.right.no == no) {
			this.right = null;
			return;
		}
		//4.我們就需要向左子樹進行遞歸刪除
		if(this.left != null) {
			this.left.delNode(no);
		}
		//5.則應當向右子樹進行遞歸刪除
		if(this.right != null) {
			this.right.delNode(no);
		}
	}

//在 BinaryTree 類增加方法

//刪除結點
	public void delNode(int no) {
		if(root != null) {
			//如果只有一個root結點, 這裡立即判斷root是不是就是要刪除結點
			if(root.getNo() == no) {
				root = null;
			} else {
				//遞歸刪除
				root.delNode(no);
			}
		}else{
			System.out.println("空樹,不能刪除~");
		}
	}

//在 BinaryTreeDemo 類增加測試代碼:

//測試一把刪除結點
		System.out.println("刪除前,前序遍歷");
		binaryTree.preOrder(); //  1,2,3,5,4
		binaryTree.delNode(5);
		//binaryTree.delNode(3);
		System.out.println("刪除後,前序遍歷");
		binaryTree.preOrder(); // 1,2,3,4

代碼運行如圖:

這篇博客是我在B站看韓順平老師數據結構和演算法的課時的筆記,記錄一下,防止忘記,也希望能幫助各位朋友。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.1 資料庫系統概述: 1.1.1資料庫的4個基本概念 資料庫的四個基本概念 - 數據 - 資料庫 - 資料庫管理系統 - 資料庫系統 數據:數據是資料庫中存儲的基本對象 數據是描述事物的一個符號,可以描述數字、圖形、聲音、語言等待,但都要經過數字化後存入計算器 資料庫(簡稱DB):資料庫是長期存 ...
  • 前文回顧:實現一個簡單的Database1(譯文) 譯註:cstsck在github維護了一個簡單的、類似sqlite的資料庫實現,通過這個簡單的項目,可以很好的理解資料庫是如何運行的。本文是第二篇,主要是實現資料庫的前端組件,編譯器與虛擬機部分功能 Part 2 世界上最簡單的SQL編譯器與虛擬機 ...
  • 在最新一屆國際資料庫頂級會議 ACM SIGMOD 2022 上,來自清華大學的李國良和張超兩位老師發表了一篇論文:《HTAP Database: What is New and What is Next》,並做了 《HTAP Database:A Tutorial》 的專項報告。這幾期學術分享會的 ...
  • 問題:【Chrome插件 Chrome extension 】報錯 Unchecked runtime.lastError: Could not establish connection. Receiving end does not exist. 在看一個別人插件的時候發現一個如上所述的報錯,雖然 ...
  • 以下內容為本人的學習筆記,如需要轉載,請聲明原文鏈接 微信公眾號「englyf」https://www.cnblogs.com/englyf/ 對於閉包的理解,其實可以歸納為,在創建函數時,同時創建了一個集合,這個集合是用來保存函數內的各個變數(無論是內部定義的,還是外部定義的),當調用函數時,變數 ...
  • 前言 請講下 JavaScript 中的數據類型? 前端面試中,估計大家都被這麼問過。 答:Javascript 中的數據類型包括原始類型和引用類型。其中原始類型包括 null、undefined、boolean、string、symbol、bigInt、number。引用類型指的是 Object。 ...
  • 1、Date => String 代碼 /** * 函數描述:時間格式化工具 * @param format {String} 格式(y-年,M-月,d-日,H-時[24],h-時[12],m-分,s-秒,S-毫秒(3位數),q-季度,ap,午前am/午後pm) * @returns {String ...
  • 滿漢樓項目 筆記目錄:(https://www.cnblogs.com/wenjie2000/p/16378441.html) 註意:筆記內容僅為實現該項目的基本後端功能,並不會實現可視化界面,效果都在控制台展示。 要完成的滿漢樓項目說明 滿漢樓項目功能多,界面複雜,涉及到複雜的awt和swing技 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...