二分搜索树(Java)

news2024/11/18 4:44:10

完整代码在最后


树结构:

1.树结构本身是一种天然的组织结构
2.高效

二分搜索树的基础

1、二叉树

1.和链表一样:动态存储

2.具有唯一的根

3.每个结点最多只有2个孩子,每个结点最多只有一个父亲

4.具有天然的递归结构

2、满二叉树

a. 叶子结点出现在二叉树的最底层,除叶子结点之外的其它结点都有两个孩子结点。
b. 一个层数为k 的满二叉树总结点数为:
c. 第i层上的结点数为:
d. 一个层数为k的满二叉树的叶子结点个数(也就是最后一层):

3、二叉树不一定是“满”的

4、二分搜索树(二分排序树)

1.是二叉树

2.每个结点:大于其左子树结点值,小于右子树结点的值

3.每个子树也是二分搜索树

二分搜索树的创建和部分功能:

1.创造结点:(内部类)

 //创建结点
    private class Node {
        private T ele;//值
        private Node left;//左孩子索引
        private Node right;

        public Node(T ele) {
            this.ele = ele;
        }
    }

2创造树:

private Node root;//根
    private int size;

    public BinearySearchTree(){
        this.root=null;
        this.size=0;
    }

3.获取树结点个数:

public int getSize(){
        return this.size;
    }

4.添加节点(递归操作)

 //添加
    public void add(T ele){
        //更新根
        this.root=addDG(this.root,ele);
        this.size++;
    }
    //向以root为根的二分搜索树添加元素
    private Node addDG(Node root,T ele){
        //终止条件
        if(root==null){
            return new Node(ele);
        }
        //递归操作
        if(root.ele.compareTo(ele)>0){
         root.left=  addDG(root.left,ele);
        }
        else{
            root.right=addDG(root.right,ele);
        }
        return root;
    }

5.查寻值是否存在(递归操作)

//查询
    public  boolean search(T ele){
        return searchDG(this.root,ele);
    }
    private boolean searchDG(Node root,T ele){
        //终止
        if(root==null){
            return false;
        }
        if(root.ele.compareTo(ele)==0){
            return true;
        }else if(root.ele.compareTo(ele)>0){
            return searchDG(root.left,ele);
        }else{
            return searchDG(root.right,ele);
        }
    }

6.遍历(先序,中序,后续)

先序:

 public void preTravel() {
        //先序
        List<T> list = new ArrayList<>();
        preTrvelDG(this.root, list);
        String res=list.stream().map(item -> item.toString()).collect(Collectors.joining("-"));
        System.out.println(res);
    }

    private void preTrvelDG(Node root,List<T>list){
        //先序
        if(root==null){
            //判断
            return ;
        }
        //操作
        list.add(root.ele);
        preTrvelDG(root.left,list);
        preTrvelDG(root.right,list);
    }

中序:

public void midTravel() {
        //中序
        List<T> list = new ArrayList<>();
        midTrvelDG(this.root, list);
        String res=list.stream().map(item -> item.toString()).collect(Collectors.joining("-"));
        System.out.println(res);
    }

    private void midTrvelDG(Node root,List<T>list){
        //中序
        if(root==null){
            //判断
            return ;
        }
        //操作
        midTrvelDG(root.left,list);
        list.add(root.ele);
        midTrvelDG(root.right,list);
    }

后续

public void suffiuxTravel() {
        //后序
        List<T> list = new ArrayList<>();
        suffiuxTrvelDG(this.root, list);
        String res=list.stream().map(item -> item.toString()).collect(Collectors.joining("-"));
        System.out.println(res);
    }

    private void suffiuxTrvelDG(Node root,List<T>list){
        //后序
        if(root==null){
            //判断
            return ;
        }
        //操作
        suffiuxTrvelDG(root.left,list);
        suffiuxTrvelDG(root.right,list);
        list.add(root.ele);
    }

7.完整代码

package com.ffyc.learn.binearysearch;

import com.ffyc.leetcodetest.sortList077;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.stream.Collectors;

//二分搜索树
public class BinearySearchTree<T extends Comparable<T>> {

    //创建结点
    private class Node {
        private T ele;//值
        private Node left;//左孩子索引
        private Node right;

        public Node(T ele) {
            this.ele = ele;
        }
    }

    private Node root;//根
    private int size;

    public BinearySearchTree(){
        this.root=null;
        this.size=0;
    }


    public int getSize(){
        return this.size;
    }
    //添加
    public void add(T ele){
        //更新根
        this.root=addDG(this.root,ele);
        this.size++;
    }
    //向以root为根的二分搜索树添加元素
    private Node addDG(Node root,T ele){
        //终止条件
        if(root==null){
            return new Node(ele);
        }
        //递归操作
        if(root.ele.compareTo(ele)>0){
         root.left=  addDG(root.left,ele);
        }
        else{
            root.right=addDG(root.right,ele);
        }
        return root;
    }
    //查询
    public  boolean search(T ele){
        return searchDG(this.root,ele);
    }
    private boolean searchDG(Node root,T ele){
        //终止
        if(root==null){
            return false;
        }
        if(root.ele.compareTo(ele)==0){
            return true;
        }else if(root.ele.compareTo(ele)>0){
            return searchDG(root.left,ele);
        }else{
            return searchDG(root.right,ele);
        }
    }
    //遍历
    public void preTravel() {
        //先序
        List<T> list = new ArrayList<>();
        preTrvelDG(this.root, list);
        String res=list.stream().map(item -> item.toString()).collect(Collectors.joining("-"));
        System.out.println(res);
    }

    private void preTrvelDG(Node root,List<T>list){
        //先序
        if(root==null){
            //判断
            return ;
        }
        //操作
        list.add(root.ele);
        preTrvelDG(root.left,list);
        preTrvelDG(root.right,list);
    }
    public void midTravel() {
        //中序
        List<T> list = new ArrayList<>();
        midTrvelDG(this.root, list);
        String res=list.stream().map(item -> item.toString()).collect(Collectors.joining("-"));
        System.out.println(res);
    }

    private void midTrvelDG(Node root,List<T>list){
        //中序
        if(root==null){
            //判断
            return ;
        }
        //操作
        midTrvelDG(root.left,list);
        list.add(root.ele);
        midTrvelDG(root.right,list);
    }
    public void suffiuxTravel() {
        //后序
        List<T> list = new ArrayList<>();
        suffiuxTrvelDG(this.root, list);
        String res=list.stream().map(item -> item.toString()).collect(Collectors.joining("-"));
        System.out.println(res);
    }

    private void suffiuxTrvelDG(Node root,List<T>list){
        //后序
        if(root==null){
            //判断
            return ;
        }
        //操作
        suffiuxTrvelDG(root.left,list);
        suffiuxTrvelDG(root.right,list);
        list.add(root.ele);
    }

    public static void main(String[] args) {
        BinearySearchTree<Integer>bst=new BinearySearchTree<>();
        Random random=new Random();
        for(int i=0;i<10;i++){
            bst.add(random.nextInt(100));
        }
        System.out.println("先");
        bst.preTravel();
        System.out.println("中");
        bst.midTravel();
        System.out.println("后");
        bst.suffiuxTravel();
    }
}

8.测试结果:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1407271.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

关于IP地址欺骗的知识,看这篇文章就差不多了

无论是什么媒介,身份盗窃始终是一种威胁。所谓的“IP欺骗”是恶意用户为了他们的黑客攻击企图快速获得可信度的一种常见方式。 考虑到每台计算机和服务器都有一个唯一的标识符(Internet Protocol或IP地址),几乎任何使用互联网的人都可能受到攻击。IP欺骗是一种“伪造”源地…

开始读 Oracle PL/SQL Programming 第6版

最近觉得PL/SQL越来越重要&#xff0c;因为这本书早就在待读列表中&#xff0c;因此决定系统的学一下。 2024年1月24日晚开始读。 在亚马逊上的评价还不错&#xff1a; 本书的第一作者是Steven Feuerstein&#xff0c;是Oracle资深的Developer Advocate。 本书的示例代码可…

Gradlew安装配置和使用

官网 https://gradle.org/install/ 在线安装 $ sdk install gradle 8.5Homebrew is “the missing package manager for macOS”. $ brew install gradle手动安装 安装包下载 安装 $ mkdir /opt/gradle $ unzip -d /opt/gradle gradle-8.5-bin.zip $ ls /opt/gradle/gradle…

使用Burp Collaborator验证无回显的RCE漏洞

使用Burp Collaborator验证无回显的RCE漏洞 1.概述2.Collaborator演示3.通过DNS查找外带命令执行结果1.概述 当应用程序容易受到命令注入攻击,但命令是异步执行时,就会发生异步操作系统命令注入漏洞。这意味着它对应用程序的响应没有明显影响 Burp Collaborator 可以帮助您…

JAVA_ArrayList添加元素时的源码分析(jdk17)

目录 ArrayList 在 Collection 中的位置&#xff1a; ArrayList 集合底层原理&#xff1a; 先总结&#xff1a; ArrayList 底层是数组结构的&#xff1a;查找快&#xff0c;增删慢 看源码&#xff1a; 看一些重要的源码&#xff1a; 第一次存元素&#xff1a; 逻辑总览…

C++ STL之queue的使用及模拟实现

文章目录 1. 介绍2. 队列的使用3. 队列的模拟实现 1. 介绍 英文解释&#xff1a; 也就是说&#xff1a; 队列是一种容器适配器&#xff0c;专门用于在FIFO上下文(先进先出)中操作&#xff0c;其中从容器一端插入元素&#xff0c;另一端提取元素。 队列作为容器适配器实现&…

go语言(十七)----json

1、结构体转json package mainimport ("encoding/json""fmt" )type Movie struct{Title string json:"title"Year int json:"year"Price int json:"rmb"Actors []string json:"actors" }func main() {movie : Mo…

数字与数学高频问题(算法村第十三关白银挑战)

数组实现加法专题 数组实现整数加法 66. 加一 - 力扣&#xff08;LeetCode&#xff09; 给定一个由 整数 组成的 非空 数组所表示的非负整数&#xff0c;在该数的基础上加一。 最高位数字存放在数组的首位&#xff0c; 数组中每个元素只存储单个数字。 你可以假设除了整数…

【刷题】 leetcode 面试题 01.06 字符串压缩

字符串压缩 字符串压缩思路一&#xff08;双指针顺畅版&#xff09;思路二&#xff08;sprintf函数巧解版&#xff09; Thanks♪(&#xff65;ω&#xff65;)&#xff89;谢谢阅读下一篇文章见&#xff01;&#xff01;&#xff01; 字符串压缩 来看题目&#xff1a; 根据题目…

OpenCV第 2 课 OpenCV 环境搭建

文章目录 第 2 课 OpenCV 环境搭建1.安装 Numpy2.从 Ubuntu 存储库安装 OpenCV3.验证 OpenCV 安装 第 2 课 OpenCV 环境搭建 1.安装 Numpy 每一张图像都有很多个像素点&#xff0c;这也导致了程序中会涉及大量的数组处理。Numpy 是一个 Python 的拓展库&#xff0c;它对多维数…

Linux环境docker安装Neo4j,以及Neo4j新手入门教学(超详细版本)

目录 1、 图数据库Neo4j简介1.1 什么是图数据库1.2 能解决什么痛点1.3 对比关系型数据库1.4 什么是Neo4j1.5 Neo4j的构建元素 2. 环境搭建2.1 安装Neo4j Community Server2.2 docker 安装Neo4j Community Server2.3 Neo4j Desktop安装 3. Neo4j - CQL使用3.1 Neo4j - CQL简介3.…

C++模板与STL【常用算法】

&#x1f308;个人主页&#xff1a;godspeed_lucip &#x1f525; 系列专栏&#xff1a;C从基础到进阶 &#x1f384;1 STL常用算法&#x1f3c6;1.1 常用遍历算法&#x1f349;1.1.1 for_each&#x1f349;1.1.2 transform &#x1f3c6;1.2 常用查找算法&#x1f34b;1.2.1 f…

STL第二讲

第二讲 视频标准库源码版本&#xff1a;gnu c 2.9.1/4.9/Visual C OOP vs GP GP是将datas与methods分开&#xff0c;OOP相反&#xff1b; 为什么list不能使用全局的sort&#xff1f; 因为sort源代码&#xff1a; *(first (last - first)/2) // 此迭代器只能是随机访问迭代…

C语言快速排序(非递归)图文详解

前言&#xff1a; 上一期分析了快速排序的三种写法&#xff0c;这三种写法有一个相同点&#xff0c;都是采用递归形式来实现的&#xff0c;那么有没有非递归的方法实现呢&#xff1f;答案是当然有&#xff0c;用非递归的方法实现快速排序&#xff0c;其实可以借助数据结构中的栈…

C++笔记(二)

函数的默认参数 如果我们自己传入数据&#xff0c;就用自己的数据&#xff0c;如果没有&#xff0c;就用默认值 语法&#xff1a; 返回值类型 函数名&#xff08;形参默认值&#xff09;{} int func&#xff08;int a&#xff0c;int b20&#xff0c;int c30&#xff09;{} …

HTTP与HTTPS的工作流程

HTTP与HTTPS的工作流程 http知识点回顾1、HTTP访问的过程2、HTTP常见状态码3、HTTP 协议一共五大特点 https的工作流程1、对称加密2、非对称加密3、https工作流程 http知识点回顾 1、HTTP访问的过程 &#xff08;1&#xff09;解析url&#xff0c;获取 url 中包含的域名&…

Unity3D Pico VR 手势识别物体交互 适配 MRTK3

当前Pico已经支持手势识别了&#xff0c;但是提供的PICO Unity Integration SDK 中是没有手势和物体交互的功能&#xff0c;Unity XR Interaction Toolkit提供的手势识别物体交互对 Quest适配的挺好的&#xff0c;Pico 当前只能用指尖点触还不能对物体进行抓握以及手势控制射线…

仅使用 Python 创建的 Web 应用程序(前端版本)第05章_共通代码

前面介绍了很多,但是让我们从本章开始实现 WTS。 在本章中,我们将实现所有页面的公共部分:SessionManager、MockDB、Model 和 Application。 SessionManger 与 Streamlit 会话交互。 在WTS中,我们的目标是使代码清晰,SessionManager负责读写st.session_state数据。 Mock…

数据结构:3_栈和队列

栈和队列 一.栈 1. 栈的概念及结构 栈&#xff1a;一种特殊的线性表&#xff0c;其只允许在固定的一端进行插入和删除元素操作。**进行数据插入和删除操作的一端称为栈顶&#xff0c;另一端称为栈底。**栈中的数据元素遵守后进先出LIFO&#xff08;Last In First Out&#x…

【QML-Qt Design Studio】

QML编程指南 ■ Qt Design Studio &#xff08;Qt Quick UI设计工具&#xff09;■ 安装Qt Design Studio■ ■ Qt Design Studio &#xff08;Qt Quick UI设计工具&#xff09; Qt Design Studio是一个用于创建酷炫、优美UI的工具。 简单概括其功能就是让UI设计转换为qml&…