`
zhouYunan2010
  • 浏览: 207839 次
  • 性别: Icon_minigender_1
  • 来自: 湖南
社区版块
存档分类

二叉排序树的实现

阅读更多
本文将讲述二叉排序树的插入和删除的原理及实现

package com.utils;

import java.util.Iterator;
import java.util.NoSuchElementException;

/**
 * 二叉排序树,也可以成为二叉查找树
 * 它的性质如下:
 * 1.若它的左子树不为空,则左子树上所有的节点均小于其根节点
 * 2.若它的右子树不为空,则右子树上所有的节点的值均大于根节点
 * 3.它的左右子树也分别为二叉排序树
 * 
 * 简单起见,假设树中元素都实现了Comparable接口或者他们可以按自然顺序比较
 */
public class BinarySortTree<E> {
	
	/**
	 * 根节点
	 */
	private Entry<E> root = null;
	
	/**
	 * 树中元素个数
	 */
	private int size = 0;
	
	
	public BinarySortTree(){
		
	}
	
	
	public int size(){
		return size;
	}
	
	public E getRoot(){
		return root==null?null:root.element;
	}
	
	/**
	 * 查找指定元素element是否在树中存在,如果查找失败确认其添加的位置
	 * 查找成功直接返回
	 * 
	 * @param t  表示从此节点开始往下查找
	 * @param f  保存t的父节点
	 * @param p  若查找成功p指向此数据元素节点,否则返回查找路径上的最后一个节点
	 * 
	 * 这是递归实现
	 */
	private boolean searchBST(Entry<E> t,Object element,Entry<E> f,Entry<E> p){
		if(t == null){
			p = f;
			return false;
		}
		Comparable<? super E> e = (Comparable<? super E>) element;
		int cmp = e.compareTo(t.element);
		if(cmp < 0){
			return searchBST(t.left,element,t,p);
		}else if(cmp > 0){
			return searchBST(t.right,element,t,p);
		}else{
			p = t;
			return true;
		}
	}
	
	
	/**
	 * 
	 * 这是非递归实现
	 */
	private boolean searchBST(Object element,Entry[] p){
		Comparable<? super E> e = (Comparable<? super E>) element;
		Entry<E> parent = root;
		Entry<E> pp = null;		//保存parent父节点
		while(parent != null){
			int cmp = e.compareTo(parent.element);
			pp = parent;
			if(cmp < 0){
				parent = parent.left;
			}else if(cmp > 0){
				parent = parent.right;
			}else{
				p[0] = parent;
				return true;
			}
		}
		p[0] = pp;
		return false;
	}
	
	/**
	 * 首先查找二叉排序树,如果找不到指定元素
	 * 则插入到二叉树中
	 */
	public boolean add(E element){
		Entry<E> t = root;
		if(t == null){		//如果根节点为空
			root = new Entry<E>(element,null);
			size = 1;
			return false;
		}
		Comparable<? super E> e = (Comparable<? super E>) element;
		Entry[] p = new Entry[1] ;
		if(!searchBST(element,p)){	//查找失败,插入元素
			Entry<E> s = new Entry<E>(element,p[0]);
			int cmp = e.compareTo((E) p[0].element);
			if(cmp < 0){
				p[0].left = s;
			}
			if(cmp > 0){
				p[0].right = s;
			}
			size++;
			return true;
		}
		return false;
	}
	
	/**
	 * 移除节点,同时调整二叉树使之为二叉排序树
	 * 实现原理:
	 * 假设要删除的节点为p,其父节点为f,而p是f的左节点
	 * 分三种情况讨论:
	 * 1.若p为叶子节点,直接删除
	 * 2.若p有只有一个左孩子或者一个右孩子,则删除p,使PL或者PR为f的左子树
	 * 3.若p的左右子树均不为空,由二叉排序树的特点可知在删除p前,中序遍历此二叉树
	 *   可以得到一个有序序列,在删去p后为了保持其他元素的相对位置不变,可以这样做:
	 *   令p的直接前驱(或直接后继)替代p,然后删除其直接前驱或直接后继。其直接前驱可由
	 *   中序遍历的特点获得
	 */
	public boolean remove(Object o){
		Entry[] p = new Entry[1] ;
		if(searchBST(o,p)){	//查找成功,删除元素
			deleteEntry(p[0]);
			return true;
		}
		return false;
	}
	
	private void deleteEntry(Entry<E> p){
		size --;
		//如果p左右子树都不为空,找到其直接后继,替换p
		if (p.left != null && p.right != null) {
			 Entry<E> s = successor(p);
			 p.element = s.element;
			 p = s;
		}
		Entry<E> replacement = (p.left != null ? p.left : p.right);

        if (replacement != null) {		//如果其左右子树有其一不为空
            replacement.parent = p.parent;
            if (p.parent == null)	//如果p为root节点
                root = replacement;
            else if (p == p.parent.left)	//如果p是左孩子
                p.parent.left  = replacement;	
            else							//如果p是右孩子
                p.parent.right = replacement;

            p.left = p.right = p.parent = null;		//p的指针清空

        } else if (p.parent == null) { // 如果全树只有一个节点
            root = null;
        } else {  //左右子树都为空,p为叶子节点
            if (p.parent != null) {
                if (p == p.parent.left)
                    p.parent.left = null;
                else if (p == p.parent.right)
                    p.parent.right = null;
                p.parent = null;
            }
        }
		
		
	}
	
	/**
	 * 返回以中序遍历方式遍历树时,t的直接后继
	 */
	static <E> Entry<E> successor(Entry<E> t) {
		if (t == null)
			return null;
		else if (t.right != null) {	//往右,然后向左直到尽头
			Entry<E> p = t.right;
			while (p.left != null)
				p = p.left;
			return p;
		} else {		//right为空,如果t是p的左子树,则p为t的直接后继
			Entry<E> p = t.parent;
			Entry<E> ch = t;
			while (p != null && ch == p.right) {	//如果t是p的右子树,则继续向上搜索其直接后继
				ch = p;
				p = p.parent;
			}
			return p;
		}
    }
	
	public Iterator<E> itrator(){
		return new BinarySortIterator();
	}
	
	//返回中序遍历此树的迭代器
	private class BinarySortIterator implements Iterator<E>{
		Entry<E> next;
	    Entry<E> lastReturned;
	    
	    public BinarySortIterator(){
	    	Entry<E> s = root;
	    	if(s !=null){
	    		while(s.left != null){	//找到中序遍历的第一个元素
	    			s = s.left;
	    		}
	    	}
	    	next = s;	//赋给next
	    }
	    
		@Override
		public boolean hasNext() {
			return next!=null;
		}

		@Override
		public E next() {
			Entry<E> e = next;
			if (e == null)
				throw new NoSuchElementException();
			next = successor(e);
			lastReturned = e;
			return e.element;
		}

		@Override
		public void remove() {
            if (lastReturned == null)
                throw new IllegalStateException();
            // deleted entries are replaced by their successors
            if (lastReturned.left != null && lastReturned.right != null)
                next = lastReturned;
            deleteEntry(lastReturned);
            lastReturned = null;
		}
	}
	
	/**
	 * 树节点,为方便起见不写get,Set方法
	 */
	static class Entry<E>{
		E element;
		Entry<E> parent;
		Entry<E> left;
		Entry<E> right;
		
		public Entry(E element,Entry<E> parent){
			this.element = element;
			this.parent = parent;
		}
		
		public Entry(){}
	}
	
	//just for test
	public static void main(String[] args) {
		BinarySortTree<Integer> tree = new BinarySortTree<Integer>();
		tree.add(45);
		tree.add(24);
		tree.add(53);
		tree.add(45);
		tree.add(12);
		tree.add(90);
		
		System.out.println(tree.remove(400));
		System.out.println(tree.remove(45));
		System.out.println("root="+tree.getRoot());
		Iterator<Integer> it = tree.itrator();
		while(it.hasNext()){
			System.out.println(it.next());
		}
		System.out.println(tree.size());
	}

}

分享到:
评论

相关推荐

    二叉排序树实现教师成果管理系统源码

    除了二叉排序树的实现,此系统还集成了基本的文件操作,这是为了将教师成果数据持久化存储。文件操作通常包括: - **数据保存**:将当前二叉排序树中的所有节点信息序列化,写入到磁盘文件中,以便下次启动系统时能...

    二叉排序树实现

    VC6.0环境下,使用C++/C编写,实现了二叉排序树的基本功能。

    数据结构实验之二叉排序树

    在"数据结构实验之二叉排序树"中,我们主要关注的是如何利用二叉排序树实现动态查找、插入和删除操作。动态查找是指在数据结构中搜索特定值的过程,对于二叉排序树,这个过程通常从根节点开始,如果目标值小于当前...

    二叉排序树 学生管理系统

    二叉排序树实现的学生管理 有创建插入 删除 查找等功能

    Java实现二叉排序树

    在Java中实现二叉排序树,我们通常会定义一个`Node`类来表示树的节点,它包含键、值以及左右子节点的引用。例如: ```java class Node { int key; Object value; Node left, right; public Node(int item) { ...

    基于二叉排序树的电话本C语言系统

    本文档将详细介绍一个基于二叉排序树(Binary Search Tree, BST)实现的电话本系统。该系统采用C语言编写,主要功能包括:添加联系人、查找联系人、删除联系人以及更新联系人信息等。通过使用二叉排序树的数据结构,...

    二叉排序树C实现代码

    二叉排序树C语言的简单实现,包含如下操作: 1.创建二叉排序树; 2.销毁二叉排序树; 3.清空二叉排序树; 4.插入指点结点至二叉排序树; 5.删除二叉排序树指点结点; 5.获取二叉排序树指定结点; 6.获取二叉排序树根...

    二叉排序树C++实现

    二叉排序树的C++实现可能涉及类的定义,包括节点结构、插入、删除、搜索和遍历的方法。文件"Binary sort Tree"可能包含了这些实现细节,具体代码会展示如何创建和操作二叉排序树。通过阅读和理解这些代码,你可以...

    二叉排序树实现 数据结构 严蔚敏

    二叉排序树实现 数据结构 c++ 严蔚敏 完全是课本上的

    二叉排序树插入

    无论是哪种实现方式,都必须确保在插入过程中不违反二叉排序树的性质,并能够处理各种特殊情况,如树为空、树中已存在待插入元素等。 实现二叉排序树插入算法时,还需要特别注意几个关键点。首先,二叉排序树节点的...

    二叉排序树_17281183_刘梦婷_leavingtbk_二叉排序树_

    在给定的压缩文件"二叉排序树_17281183_刘梦婷"中,可能包含了实现这些功能的代码示例或练习题目,用于帮助学习者理解和掌握二叉排序树的操作。 通过理解和熟练运用二叉排序树,我们可以有效地处理各种数据结构问题...

    数据结构课程设计二叉排序树的实现

    ### 数据结构课程设计二叉排序树的实现 #### 一、二叉排序树的基本概念 二叉排序树(Binary Search Tree, BST),又称二叉查找树或有序二叉树,是一种特殊的二叉树,其特点在于对于任意节点而言,其左子树中的所有...

    数据结构课程设计 二叉排序树与平衡二叉排序树

    在这个课程设计中,我们将重点探讨如何利用二叉链表作为存储结构来实现二叉排序树,并进一步研究平衡二叉排序树的概念。 首先,二叉排序树的特性是每个节点的左子树只包含小于该节点的元素,而右子树则包含大于或...

    二叉排序树与文件操作

    【二叉排序树与文件操作】 功能要求: (1)从键盘输入一组学生记录建立二叉排序树; (2)二叉排序树存盘; (3)由文件恢复内存的二叉排序树; (4)中序遍历二叉排序树; (5)求二叉排序树深度; (6)求二叉...

    实现二叉排序树的各种算法

    在"实现二叉排序树的各种算法"的实验中,通常会涉及以下关键算法: 1. **树的构建**:从一组有序或无序的数据中构建二叉排序树。这可以通过遍历数据并按照二叉排序树的规则逐个插入节点来实现。 2. **插入操作**:...

    数据结构课程设计实现二叉排序树第九组数据结构课程设计二叉排序树实现

    在实现二叉排序树的过程中,首先需要明确设计任务。根据描述,设计任务可能包括理解二叉排序树的概念,分析问题和需求,然后进行概要设计和详细设计。概要设计阶段,我们需要考虑如何存储二叉树的节点,以及设计用户...

    数据结构之二叉排序树的生成

    在压缩包文件“二叉排序树的相关操作”中,可能包含了实现这些功能的源代码,如C、C++或Java等编程语言的实现。通过阅读和分析这些代码,我们可以更深入地理解二叉排序树的内部机制,掌握数据结构和算法在实际编程中...

Global site tag (gtag.js) - Google Analytics