二叉搜索树和二叉树的代码重构

BST继承自二叉树

在这里插入图片描述

代码

二叉树

import java.util.LinkedList;
import java.util.Queue;
@SuppressWarnings("unchecked")
// 实现BinaryTreeInfo接口是为了使用打印二叉树的工具,非必须
public class BinaryTree<E> implements BinaryTreeInfo {
	protected int size; // 元素数量
	protected Node<E> root; // 根节点
	
	/**
	 * 访问器接口 ——> 访问器抽象类
	 * 增强遍历接口
	 */
	/*public static interface Visitor<E>{
		void visit(E element);
	}*/
	public static abstract class Visitor<E> {
		boolean stop;
		// 如果返回true,就代表停止遍历
		public abstract boolean visit(E element);
	}
	/**
	 * 内部类,节点类
	 */
	public static class Node<E> {
		E element; // 元素值
		Node<E> left; // 左节点
		Node<E> right; // 右节点
		Node<E> parent; // 父节点

		public Node(E element, Node<E> parent) {
			this.element = element;
			this.parent = parent;
		}
		public boolean isLeaf(){ // 是否叶子节点
			return left==null && right==null;
		}
		public boolean hasTwoChildren(){ // 是否有两个子节点
			return left!=null && right!=null;
		}
	}
	/**
	 * 元素的数量
	 */
	public int size() {
		return size;
	}
	/**
	 * 是否为空
	 */
	public boolean isEmpty() {
		return size == 0;
	}
	/**
	 * 清空所有的元素
	 */
	public void clear() {
		root = null;
		size = 0;
	}
	/**
	 * 前序遍历
	 */
	public void preorder(Visitor<E> visitor){
		if (visitor == null) return;
		preorder(root, visitor);
	}
	public void preorder(Node<E> node, Visitor<E> visitor){
		if(node == null || visitor.stop) return;
		// 根
		visitor.stop = visitor.visit(node.element);
		// 左
		preorder(node.left, visitor);
		// 右
		preorder(node.right, visitor);
	}
	/**
	 * 中序遍历
	 * @param visitor
	 */
	public void inorder(Visitor<E> visitor){
		if (visitor == null) return;
		inorder(root, visitor);
	}
	public void inorder(Node<E> node, Visitor<E> visitor){
		if(node == null || visitor.stop) return;
		// 左
		inorder(node.left, visitor);
		// 根
		if (visitor.stop) return;
		visitor.stop = visitor.visit(node.element);
		// 右
		inorder(node.right, visitor);
	}
	/**
	 * 后序遍历
	 */
	public void postorder(Visitor<E> visitor){
		if(visitor == null) return;
		postorder(root, visitor);
	}
	public void postorder(Node<E> node, Visitor<E> visitor){
		if(node == null || visitor.stop) return;
		// 左
		postorder(node.left, visitor);
		// 右
		postorder(node.right, visitor);
		// 根
		if(visitor.stop) return;
		visitor.stop = visitor.visit(node.element);
	}
	/**
	 * 层次遍历
	 */
	public void levelOrder(Visitor<E> visitor){
		if(root == null || visitor.stop) return;
		Queue<Node<E>> queue = new LinkedList<>(); // 队列
		queue.offer(root);
		
		while(!queue.isEmpty()){
			Node<E> node = queue.poll();
			if(visitor.visit(node.element)) return;
			
			if(node.left != null) {
				queue.offer(node.left);
			}
			if(node.right != null) {
				queue.offer(node.right);
			}
		}
	}
	/**
	 * 高度(递归)
	 */
	public int height1(){
		return height1(root);
	}
	public int height1(Node<E> node){
		if(node == null) return 0;
		return 1 + Math.max(height1(node.left), height1(node.right));
	}
	/**
	 * 高度(迭代)
	 */
	public int height(){
		if(root == null) return 0;
		int levelSize = 1; // 存储每一层的元素数量	
 		int height = 0; // 树的高度
		Queue<Node<E>> queue = new LinkedList<>();
		queue.offer(root);
		
		while(!queue.isEmpty()){	
			Node<E> node = queue.poll();
			levelSize--;
			if(node.left != null) {
				queue.offer(node.left);
			}
			if(node.right != null) {
				queue.offer(node.right);
			}
			if(levelSize == 0){ // 即将要访问下一层
				levelSize = queue.size();
				height++;
			}
		}
		return height;
	}
	/**
	 * 是否是完全二叉树
	 */
	public boolean isComplete(){
		if(root == null){
			return false;
		}
		Queue<Node<E>> queue = new LinkedList<>();
		queue.offer(root);
		
		boolean leaf = false;
		while(!queue.isEmpty()){
			Node<E> node = queue.poll();
			if(leaf && !node.isLeaf()){ // 要求是叶子结点,但是当前节点不是叶子结点
				return false;
			}
			if(node.left != null){
				queue.offer(node.left);
			}else if(node.right != null){
				// node.left==null && node.right!=null
				return false;
			}
			if(node.right != null){
				queue.offer(node.right);
			}else{
				// node.left==null && node.right==null
				// node.left!=null && node.right==null
				leaf = true; // 要求后面都是叶子节点
			}
		}
		return true;
	}
	/**
	 * 前驱节点:中序遍历时的前一个节点
	 * 求前驱节点
	 */
	protected Node<E> predecessor(Node<E> node) {
		if(node == null) return null;
		
		// 前驱节点在左子树中(left.right.right.right....)
		if(node.left != null ){ // 左子树不为空,则找到它的最右节点
			Node<E> p = node.left;	
			while(node.right != null){
				p = p.right;
			}
			return p;
		}
		// 能来到这里说明左子树为空
		// 当父节点不为空,则顺着父节点找,直到找到【某结点为父节点的右子节点】时
		while(node.parent != null && node.parent.left==node){
			node = node.parent;
		}
		// 来到这里有以下两种情况:
		// node.parent == null	无前驱,说明是根结点
		// node.parent.right == node 找到【某结点为父节点的右子节点】
		return node.parent;
	}
	/**
	 * 后继节点:中序遍历时的后一个节点
	 * 求后继节点
	 */
	protected Node<E> successor(Node<E> node) {
		if(node == null) return null;
		// 后继节点与前驱节点正好相反
		
		// 后继节点在右子树中(node.right.left.left...)
		if(node.right != null){
			Node<E> p = node.right;
			while(p.left != null){
				p = p.left;
			}
			return p;
		}
		// 来到这里说明没有右节点
		// 当父节点不为空,则顺着父节点找,直到找到【某结点为父节点的左子节点】时
		while(node.parent!=null && node.parent.right==node){
			node = node.parent;
		}
		
		// 来到这里有以下两种情况:
		// node.parent == null   无前驱,说明是根结点
		// node.parent.left == node  找到【某结点为父节点的左子节点】
		return node.parent;
	}
	/**
	 * BinaryTreeInfo 工具,用来打印二叉树
	 */
	@Override
	public Object root() {
		return root;
	}
	@Override
	public Object left(Object node) {
		return ((Node<E>)node).left;
	}
	@Override
	public Object right(Object node) {
		return ((Node<E>)node).right;
	}
	@Override
	public Object string(Object node) {
		Node<E> myNode = (Node<E>)node;
		String parentStr = "null";
		if(myNode.parent != null){
			parentStr = myNode.parent.element.toString();
		}
		return myNode.element + "_p(" + parentStr + ")";
	}
}

二叉搜索树

import java.util.Comparator;

@SuppressWarnings("unchecked")
public class BinarySearchTree<E> extends BinaryTree<E> {
	
	// 比较器,根据传入的比较器实现 compareTo() 方法
	private Comparator<E> comparator;
	
	public BinarySearchTree (Comparator<E> comparator){ // 可以传一个比较器
		this.comparator = comparator;
	}
	public BinarySearchTree(){ // 不传比较器,相当于传入一个 null
		this(null); //
	}
	
	/**
	 * 添加元素
	 */
	public void add(E element) {
		elementNotNullCheck(element); // 不能传入空节点
		// 传入第一个节点
		if(root == null){
			root = new Node<>(element, null);
			size++;
			return;
		}
		Node<E> node = root;
		Node<E> parent = root;
		int cmp = 0;
		while(node != null){
			parent = node; // 父节点
			cmp = compareTo(node.element, element); // 方向
			if(cmp < 0){
				node = node.right;
			}else if(cmp > 0){
				node = node.left;
			}else{ // 相等,最好是覆盖掉,也可以采取其他操作,看具体需求
				node.element = element;
				return;
			}
		}
		Node<E> newNode = new Node<>(element, parent);
		if(cmp < 0){
			parent.right = newNode;
		}else{
			parent.left = newNode;
		}
		size++;
	}
	/**
	 * 是否包含某元素
	 */
	public boolean contains(E element) {
		return node(element) != null;
	}
	/**
	 * 根据传入的值删除元素
	 */
	public void remove(E element) {
		remove(node(element));
	}
	// 根据节点删除元素
	private void remove(Node<E> node) {
		if (node == null) return;
		
		size--;
		
		if (node.hasTwoChildren()) { // 度为2的节点
			// 找到后继节点
			Node<E> s = successor(node);
			// 用后继节点的值覆盖度为2的节点的值
			node.element = s.element;
			// 删除后继节点
			node = s;
		}
		
		// 删除node节点(node的度必然是1或者0)
		Node<E> replacement = node.left != null ? node.left : node.right;
		
		if (replacement != null) { // node是度为1的节点
			// 更改parent
			replacement.parent = node.parent;
			// 更改parent的left、right的指向
			if (node.parent == null) { // node是度为1的节点并且是根节点
				root = replacement;
			} else if (node == node.parent.left) {
				node.parent.left = replacement;
			} else { // node == node.parent.right
				node.parent.right = replacement;
			}
		} else if (node.parent == null) { // node是叶子节点并且是根节点
			root = null;
		} else { // node是叶子节点,但不是根节点
			if (node == node.parent.left) {
				node.parent.left = null;
			} else { // node == node.parent.right
				node.parent.right = null;
			}
		}
	}
	
	// 根据元素值获取节点元素
	private Node<E> node(E element){
		elementNotNullCheck(element);
		
		Node<E> node = root;
		while(node != null){
			int cmp = compareTo(element, node.element);
			if(cmp < 0){
				node = node.left;
			}else if (cmp > 0){
				node = node.right;
			}else{ // cmp == 0
				return node;
			}
		}
		return null;
	}
	// 节点元素比较
	private int compareTo(E e1, E e2) {
		if (comparator != null) { // 传入比较器则通过比较器比较
			return comparator.compare(e1, e2);
		}
		// 没传比较器,元素内部必须自行实现了 Comparable 接口
		return ((Comparable<E>)e1).compareTo(e2);
	}
	// 检测传入的节点是否为空
	private void elementNotNullCheck(E element) {
		if (element == null) { // 不能传入空节点
			throw new IllegalArgumentException("element must not null");
		}
	}
	
}

Main类main函数测试

import java.util.Comparator;

public class Main {
	
	public static void main(String[] args) {
		test2();
	}
	
	
	public static class PersonComparator implements Comparator<Person> { // 比较器
		@Override
		public int compare(Person e1, Person e2) {
			return e1.getAge() - e2.getAge();
		}
	}
	public static class PersonComparator2 implements Comparator<Person> {
		@Override
		public int compare(Person e1, Person e2) {
			return e2.getAge() - e1.getAge();
		}
	}
	
	// Integer类型的数据
	public static void test1(){
		Integer date[] = new Integer[] { 7, 4, 9, 2, 5, 8, 11, 3, 12, 1};
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		for (int i = 0; i < date.length; i++) {
			bst.add(date[i]);
		}
		BinaryTrees.println(bst);
	}

	// Person类型的数据
	public static void test2(){
		// Java,匿名类
		BinarySearchTree<Person> bst = new BinarySearchTree<>(new Comparator<Person>() {
			@Override
			public int compare(Person e1, Person e2) {
				return e2.getAge() - e1.getAge();
			}
		});
		Integer date[] = new Integer[] { 7, 4, 9, 2, 5, 8, 11, 3, 12, 1};
		for (int i = 0; i < date.length; i++) {
			bst.add(new Person(date[i]));
		}
		BinaryTrees.println(bst);
	}
	
	// 保存打印结果到文件
	public static void test3(){
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		for(int i = 0; i < 40; i++){
			bst.add((int)(Math.random()*100));
		}
		String string = BinaryTrees.printString(bst);
		string += "\n";
		Files.writeToFile("F:/1.txt", string);
	}
	
	// add() 时值相等的处理
	public static void test4(){
		BinarySearchTree<Person> bst = new BinarySearchTree<>(new Comparator<Person>() {
			@Override
			public int compare(Person o1, Person o2) {
				return o1.getAge() - o2.getAge();
			}
		});
		bst.add(new Person(15, "jack"));
		bst.add(new Person(16, "rose"));
		bst.add(new Person(10, "jerry"));
		bst.add(new Person(10, "kali")); // add()时值相等最好覆盖,否则则不会替换
		BinaryTrees.println(bst);
	}
	
	// 遍历
	public static void test5(){
		Integer date[] = new Integer[] { 7, 4, 9, 2, 5, 8, 11, 3, 12, 1};
		Person persons[] = new Person[10];
		BinarySearchTree<Person> bst = new BinarySearchTree<>(new Comparator<Person>() {
			@Override
			public int compare(Person o1, Person o2) {
				return o2.getAge() - o1.getAge();
			}
		});
		
		for(int i = 0; i < persons.length; i++){
			persons[i] = new Person(date[i]);
			bst.add(persons[i]);
		}
		BinaryTrees.println(bst);
		System.out.print("层次遍历:");
		bst.levelOrder(new Visitor<Person>() {
			@Override
			public boolean visit(Person element) {
				System.out.print(element + " ");
				return false;
			}
		});
	}
	
	// 访问器遍历
	public static void test6(){
		Integer date[] = new Integer[] { 7, 4, 9, 2, 5, 8, 11, 3, 12, 1};
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		for (int i = 0; i < date.length; i++) {
			bst.add(date[i]);
		}
		BinaryTrees.print(bst);
		
		System.out.print("层次遍历:");
		bst.levelOrder(new Visitor<Integer>() {
			@Override
			public boolean visit(Integer element) {
				System.out.print(element + " ");
				return false;
			}
		});
		System.out.println();
		
		System.out.print("前序遍历:");
		bst.preorder(new Visitor<Integer>() {
			public boolean visit(Integer element) {
				System.out.print(element + " ");
//				return element == 2 ? true : false;
				return false;
			}
		});
		System.out.println();
		
		System.out.print("中序遍历:");
		bst.inorder(new Visitor<Integer>() {
			@Override
			public boolean visit(Integer element) {
				System.out.print(element + " ");
				return false;
			}
		});
		System.out.println();
		
		System.out.print("后序遍历:");
		bst.postorder(null);
		bst.postorder(new Visitor<Integer>() {
			@Override
			public boolean visit(Integer element) {
				System.out.print(element + " ");
				return false;
			}
		});
		
	}
	
	// 高度
	public static void test7(){
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		for(int i = 0; i < 20; i++){
			bst.add((int)(Math.random()*100));
		}
		BinaryTrees.print(bst);
		System.out.println();
//		System.out.println(bst.height()1);//递归求高度
		System.out.println(bst.height());
	}
	
	// 是否是完全二叉树
	public static void test8(){
		/* 
		 *    7
		 *  4   8
		 *1   5 
		 * 
		 */
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		Integer date[] = new Integer[] { 7, 4, 8, 1, 5};
		for (int i = 0; i < date.length; i++) {
			bst.add(date[i]);
		}
		BinaryTrees.println(bst);
		System.out.println(bst.isComplete());
	}

	// 是否包含某个结点
	public static void test9(){
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		Integer date[] = new Integer[] { 7, 4, 8, 1, 5};
		for (int i = 0; i < date.length; i++) {
			bst.add(date[i]);
		}
		BinaryTrees.println(bst);
		System.out.println(bst.contains(6));
	}
	
	// 删除节点
	public static void test10(){
		BinarySearchTree<Integer> bst = new BinarySearchTree<>();
		Integer date[] = new Integer[] { 7, 4, 8, 1, 5};
		for (int i = 0; i < date.length; i++) {
			bst.add(date[i]);
		}
		BinaryTrees.println(bst);
		bst.remove(8);
		BinaryTrees.println(bst);
	}
	
}

如果是用普通二叉树类用来添加,不知道添加规则,所以没有意义用二叉搜索树类的规则添加元素才有意义,在实际应用中,可自定义规则,比如当前后插入两个相同的元素时,是选择放在原来节点的右边还是左边或者是覆盖,都可自定义

二叉搜索树的复杂度

BST的添加删除搜索,比较的次数与树的高度有关,但与元素个数无关
如果二叉树是满二叉树或完全二叉树 ,复杂度=O(h)=O(logn)(其中n是节点个数,logn是比较次数)
二叉搜索树只适合于随机添加的数字,如果按顺序添加就会退化成链表。
在这里插入图片描述
删除2之后,原来根节点的左孩子也就变成了链表
在这里插入图片描述
平衡二叉树就是通过平衡因子来使二叉排序树维持平衡,从而防止退化成链表

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值