面试整理 - 二叉排序树 c语言 及java 例子

简介: 面试整理 - 二叉排序树 c语言 及java 例子

什么是二叉排序树?

二叉排序树(binary search tree,bst)是一种特殊的二叉树,其中每个节点具有一个键值,并且满足一下两个要求:

  1. 对于任何节点x,其左子树上所有节点的关键字值小于x的关键字值。

  2. 对于任何节点x,其有子树上所有节点的关键字值大于x的关键字值。

由于以上这两个性质,因此二叉排序树也被称为有序二叉树。在执行查找,**,删除等操作时,二叉排序树都可以保持平衡,使得时间复杂度变为o(log n)级别,极大的提高了操作效率。

二叉排序树原理

  1. 二叉排序树中每个结点最多有两个子树,且左子树的所有关键字(节点值)小于当前结点的关键字,右子树的所有关键字大于当前结点的关键字。
  2. 左右子树也都是二叉排序树。
  3. 当某个结点被查找时,比较待查找关键字与当前结点的关键字大小,若相等,则说明查找成功;否则根据大小关系向左或向右子树递归查找,直到找到对应的结点或者确认结点不存在。
  4. 一个新的结点时,从根结点开始将待结点与当前结点进行比较,根据大小关系向左或向右递归到叶子结点停止。然后把新结点**到该叶子结点的左或右子树上。
  5. 当删除一个结点时,按照查找的方式找到需要删除的结点,如果该结点没有子结点,则直接删除即可;如果只有左子树或右子树,则将左子树或右子树直接替换到该结点位置;如果既有左子树又有右子树,则找到该结点的前驱或后继(中序遍历时大于当前结点的最小节点或者小于当前结点的最大节点),然后用其值来替换需要删除的结点的值,并删除前驱或后继。

使用Java实现二叉树排序的示例代码:

// 节点类
class Node {
   
    int value;
    Node left, right;

    public Node(int item) {
   
        value = item;
        left = right = null;
    }
}

public class BinaryTree {
   
    Node root;

    public BinaryTree() {
   
        root = null;
    }

    // 将给定节点的值插入到二叉树中
    private Node insertNode(Node root, int value) {
   
        if (root == null) {
   
            root = new Node(value);
            return root;
        }

        if (value < root.value) {
   
            root.left = insertNode(root.left, value);
        } else if (value > root.value) {
   
            root.right = insertNode(root.right, value);
        }

        return root;
    }

    // 中序遍历,将节点的值按升序返回。
    private void traverseInOrder(Node root, List<Integer> values) {
   
        if (root != null) {
   
            traverseInOrder(root.left, values);
            values.add(root.value);
            traverseInOrder(root.right, values);
        }
    }

    // 对外的排序方法,接收一个整数数组并返回升序排列后的结果。
    public List<Integer> sort(int[] arr) {
   
        for (int i : arr) {
   
            root = insertNode(root, i);
        }

        List<Integer> sortedList = new ArrayList<>();
        traverseInOrder(root, sortedList);

        return sortedList;
    }

    // 测试代码
    public static void main(String[] args) {
   
        BinaryTree tree = new BinaryTree();
        int[] arr = {
   5, 3, 7, 1, 9};

        List<Integer> sortedList = tree.sort(arr);
        System.out.println(sortedList); // Output: [1, 3, 5, 7, 9]
    }
}

该示例中,我们使用节点类Node表示二叉树中的每个节点。二叉树类BinaryTree包含插入节点、中序遍历和排序三个方法

C 语言实现二叉搜索树排序(Binary Search Tree Sorting)的示例代码:

#include <stdio.h>
#include <stdlib.h>

// 定义二叉树节点结构体
struct Node {
   
    int data;           // 数据
    struct Node* left;  // 左子节点指针
    struct Node* right; // 右子节点指针
};

// 创建新节点
struct Node* newNode(int data) {
   
    struct Node* node = (struct Node*)malloc(sizeof(struct Node));
    node->data = data;
    node->left = NULL;
    node->right = NULL;
    return node;
}

// 插入节点
struct Node* insert(struct Node* node, int data) {
   
    if(node == NULL)
        return newNode(data);
    if(data < node->data)
        node->left = insert(node->left, data);
    else if(data > node->data)
        node->right = insert(node->right, data);
    return node;
}

// 遍历节点并按序排列输出结果 
void inorderTraversal(struct Node* node) {
   
    if(node != NULL) {
   
        inorderTraversal(node->left);
        printf("%d ", node->data);
        inorderTraversal(node->right);
    }
}

// 测试示例
int main() {
   
    int arr[] = {
   7, 5, 1, 8, 3, 6, 0, 9, 4, 2};
    int n = sizeof(arr)/sizeof(arr[0]);
    struct Node* root = NULL;

    for(int i=0; i<n; i++)
        root = insert(root, arr[i]);

    inorderTraversal(root);

    return 0;
}

以上是一个简单的二叉搜索树排序的实现,可以根据需要进行修改和优化。

相关文章
|
6天前
|
负载均衡 NoSQL 算法
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
这篇文章是关于Java面试中Redis相关问题的笔记,包括Redis事务实现、集群方案、主从复制原理、CAP和BASE理论以及负载均衡算法和类型。
一天五道Java面试题----第十天(简述Redis事务实现--------->负载均衡算法、类型)
|
6天前
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
6天前
|
存储 监控 安全
一天十道Java面试题----第三天(对线程安全的理解------>线程池中阻塞队列的作用)
这篇文章是Java面试第三天的笔记,讨论了线程安全、Thread与Runnable的区别、守护线程、ThreadLocal原理及内存泄漏问题、并发并行串行的概念、并发三大特性、线程池的使用原因和解释、线程池处理流程,以及线程池中阻塞队列的作用和设计考虑。
|
4天前
|
存储 缓存 网络协议
复盘女朋友面试4个月的Java基础题
这篇文章是关于Java基础面试题的复盘,涵盖了HashMap原理、对象序列化作用等高频面试问题,并强调了Java基础知识的重要性。
复盘女朋友面试4个月的Java基础题
|
6天前
|
存储 NoSQL Java
一天五道Java面试题----第十一天(分布式架构下,Session共享有什么方案--------->分布式事务解决方案)
这篇文章是关于Java面试中的分布式架构问题的笔记,包括分布式架构下的Session共享方案、RPC和RMI的理解、分布式ID生成方案、分布式锁解决方案以及分布式事务解决方案。
一天五道Java面试题----第十一天(分布式架构下,Session共享有什么方案--------->分布式事务解决方案)
|
6天前
|
存储 安全 Java
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
这篇文章是关于Java面试的第二天笔记,涵盖了HashMap与HashTable的区别、ConcurrentHashMap的实现原理、IOC容器的实现方法、字节码的概念和作用、Java类加载器的类型、双亲委派模型、Java异常体系、GC如何判断对象可回收、线程的生命周期及状态,以及sleep、wait、join、yield的区别等十道面试题。
一天十道Java面试题----第二天(HashMap和hashTable的区别--------》sleep、wait、join)
|
6天前
|
SQL Java 数据库连接
一天五道Java面试题----第六天(1)
这篇文章是关于Java面试中常见的五个问题,包括MyBatis和Hibernate的对比、MyBatis中#{}和${}的区别、MyBatis插件的运行原理及开发流程、索引的基本原理以及MySQL聚簇索引和非聚簇索引的区别。
|
6天前
|
缓存 NoSQL Redis
一天五道Java面试题----第九天(简述MySQL中索引类型对数据库的性能的影响--------->缓存雪崩、缓存穿透、缓存击穿)
这篇文章是关于Java面试中可能会遇到的五个问题,包括MySQL索引类型及其对数据库性能的影响、Redis的RDB和AOF持久化机制、Redis的过期键删除策略、Redis的单线程模型为何高效,以及缓存雪崩、缓存穿透和缓存击穿的概念及其解决方案。
|
6天前
|
前端开发 Java 数据库连接
一天十道Java面试题----第五天(spring的事务传播机制------>mybatis的优缺点)
这篇文章总结了Java面试中的十个问题,包括Spring事务传播机制、Spring事务失效条件、Bean自动装配方式、Spring、Spring MVC和Spring Boot的区别、Spring MVC的工作流程和主要组件、Spring Boot的自动配置原理和Starter概念、嵌入式服务器的使用原因,以及MyBatis的优缺点。
|
7天前
|
存储 Java 编译器