[数据结构]二叉树的旋转

  在很多平衡树中都用到了树的旋转来维护,比如说红黑树,以及竞赛比较常用的树堆(Treap)

树的旋转既要能改变最大深度,使得平衡树平衡又不能破坏BST(二叉查找树,Binary Search Tree)

的性质,还是比较困难。

  先不说BST的,先看看一棵普通的树是怎么旋转的(虽然是一样),然后再思考,为什么这样可以

不破坏BST的性质。

(PS,有两个结点的内容是一样,就在绘图的时候右边加了个标识)

  虽然不太好看懂,那么就仔细观察吧(笑),那么就把木有怎么变的结点

标出来

于是可以发现这几点

1)左旋前和左旋后原先根节点的左子树不会改变

2)左旋前和左旋后后来的根节点的右子树不会改变

3)右旋前和右旋后原先的根节点的右子树不会改变

4)右旋前和右旋后后来的根节点的左子树不会改变

(以上几点的前提是,你不用一维数组存树)

那再来看看改变的结点

  先说左旋吧,新的根节点是原先根节点的右子树,原来的根节点变成新的根节点

的左子树,那新的根节点的原来的左子树去哪了?成了原来的根节点的右子树

  右旋就和上面差不多了,自行琢磨一下很容易解决

  接下来讲一下BST,我们将这几个会改变的结点进行编号,从上到下,标为1,2,3

根据BST的性质应该是结点1<结点3<结点2,将结点3变为结点1的右子树满足结点1<结点3,

结点1是结点2的左子树说明,结点1<结点2,结点3<结点2。满足这个条件

  右旋和左旋仍然可以像这样理解。



Code

(我也不知道为什么突然想用java写了,如果是C/C++的同学看不懂的地方可以在评论问一下)

树的结点(PS,我养成了封装的好习惯)

TreeNode.java

package MyTree;

/**
 * 树节点
 **/
public class TreeNode<T> {

    private T data;
    private TreeNode<T> father;
    private TreeNode<T> left;
    private TreeNode<T> right;

    {
        father = null;
        left = null;
        right = null;
    }

    /**
     * 构造一个空的树节点
     **/
    public TreeNode(){

    }

    /**
     * 构造一个拥有指定数据和父节点的节点
     * @param data 指定的数据
     * @param father 父节点的引用
     **/
    public TreeNode(T data, TreeNode<T> father){
        this.data = data;
        this.father = father;
    }

    /**
     * 构造一个拥有指定数据、父节点和左右子树的<br>
     * 结点
     * @param data        指定的数据
     * @param father    父节点
     * @param left        左子树
     * @param right        右子树
     */
    public TreeNode(T data, TreeNode<T> father, TreeNode<T> left,
            TreeNode<T> right){
        this.data = data;
        this.father = father;
        this.left = left;
        this.right = right;
    }

    /**
     * 以该节点作为根节点进行左旋
     * @return 旋转成功返回true,<br>
     *             否则返回false
     */
    public synchronized boolean rotateLeft(){
        if(this.right == null)    return false;
        TreeNode<T> buf = this.right;
        TreeNode<T> f = this.father;
        this.right = buf.left;
        this.father = buf;
        buf.left = this;
        buf.father = f;
        return true;
    }

    /**
     * 以该节点作为根节点进行右旋
     * @return 旋转成功返回true,<br>
     *             否则返回false
     */
    public synchronized boolean rotateRight(){
        if(this.left == null)    return false;
        TreeNode<T> buf = this.left;
        TreeNode<T> f =    this.father;
        this.left = buf.right;
        this.father = buf;
        buf.right = this;
        buf.father = f;
        return true;
    }

    /**
     * @return the data
     */
    public T getData() {
        return data;
    }

    /**
     * @param data the data to set
     */
    public void setData(T data) {
        this.data = data;
    }

    /**
     * @return the father
     */
    public TreeNode<T> getFather() {
        return father;
    }

    /**
     * @param father the father to set
     */
    public void setFather(TreeNode<T> father) {
        this.father = father;
    }

    /**
     * @return the left
     */
    public TreeNode<T> getLeft() {
        return left;
    }

    /**
     * @param left the left to set
     */
    public void setLeft(TreeNode<T> left) {
        this.left = left;
    }

    /**
     * @return the right
     */
    public TreeNode<T> getRight() {
        return right;
    }

    /**
     * @param right the right to set
     */
    public void setRight(TreeNode<T> right) {
        this.right = right;
    }

}

接下来按照上面的数据测试一下(当然啦,我也不能确保代码是正确的,如果您看出了什么问题可以在评论中指出)

TestRotate.java

 1 package MyTree;
 2
 3 public class TestRotate {
 4
 5     protected static void preorder(TreeNode<? extends String> root){
 6         if(root == null)    return;
 7         System.out.print(root.getData());
 8         preorder(root.getLeft());
 9         preorder(root.getRight());
10     }
11
12     public static void main(String[] args) {
13
14         @SuppressWarnings("unchecked")
15         TreeNode<String> nodes[] = new TreeNode[5];
16         nodes[0] = new TreeNode<>("h", null);
17         nodes[1] = new TreeNode<>("e", nodes[0]);
18         nodes[0].setLeft(nodes[1]);
19         nodes[2] = new TreeNode<>("l", nodes[0]);
20         nodes[0].setRight(nodes[2]);
21         nodes[3] = new TreeNode<>("l", nodes[2]);
22         nodes[2].setLeft(nodes[3]);
23         nodes[4] = new TreeNode<>("o", nodes[2]);
24         nodes[2].setRight(nodes[4]);
25
26         preorder(nodes[0]);
27
28         nodes[0].rotateLeft();
29
30         System.out.println();
31
32         preorder(nodes[2]);
33
34         System.out.println();
35
36         nodes[2].rotateRight();
37
38         preorder(nodes[0]);
39
40     }
41
42 }

使用的时候请不要无视两段代码头的package,不然编译错误别怪我

时间: 2024-10-05 20:50:08

[数据结构]二叉树的旋转的相关文章

数据结构二叉树——建立二叉树、中序递归遍历、非递归遍历、层次遍历

数据结构二叉树-- 编写函数实现:建立二叉树.中序递归遍历.借助栈实现中序非递归遍历.借助队列实现层次遍历.求高度.结点数.叶子数及交换左右子树. ("."表示空子树) #include<stdio.h> #include<stdlib.h> //***********二叉树链表节点结构 typedef char DataType; typedef struct Node {  DataType data;  struct Node*LChild;  struc

POJ 3367 Expressions(数据结构-二叉树)

Expressions Description Arithmetic expressions are usually written with the operators in between the two operands (which is called infix notation). For example, (x+y)*(z-w) is an arithmetic expression in infix notation. However, it is easier to write

[数据结构] 二叉树的建立及其基本操作

如图: 代码: #include <iostream> #include <stdio.h> #include <algorithm> #include <string.h> using namespace std; char ch; typedef struct BinNode { char data; struct BinNode *lchild,*rchild; }BinNode,*BinTree; //二叉树链式存储结构 void CreateBin

数据结构--二叉树(定义与存储结构)

什么是二叉树 是具有n个节点的有限集合,由一个根节点和两棵互不相交二叉树组成.如图 从名字简单理解,就是具有2个树叉的树形结构,当然这不是绝对的,正如上图所示,我也可以只有一个树叉. 二叉树具有五种基本形态: (1)空二叉树 (2)只有一个根结点的二叉树 (3)只有左子树 (4)只有右子树 (5)既有左子树又有右子树 完全二叉树 这种二叉树,是二叉树中常用的专业术语,不是说一个完整的二叉树就是完全二叉树,这种二叉树叫满二叉树,如图 简单理解就像图片中,完全二叉树中每个节点的编号,都能映射到满二叉

(编程训练)再回首,数据结构——二叉树的前序、中序、后序遍历(非递归)

最近在复习数据结构,顺便看看大一的时候写的代码,看完之后比当初有了更加深刻的体会. 希望这些能提供给初学者一些参考. 在VC++6.0下可运行,当初还写了不少注释. 可以和(编程训练)再回首,数据结构--二叉树的前序.中序.后序遍历(递归)对比着看 [问题描述] 根据顺序存储结构建立二叉树的二叉链表,并对二叉树进行先序.中序.后序遍历. [基本要求] ·功能:根据顺序存储结构建立二叉树的二叉链表,并进行先序.中序.后序遍历. ·输入:输入二叉树的顺序存储. ·输出:二叉树的先序.中序.后序遍历序

浅谈数据结构-二叉树

浅谈数据结构-二叉树 二叉树是树的特殊一种,具有如下特点:1.每个结点最多有两颗子树,结点的度最大为2.2.左子树和右子树是有顺序的,次序不能颠倒.3.即使某结点只有一个子树,也要区分左右子树. 一.特殊的二叉树及特点 1.斜树 所有的结点都只有左子树(左斜树),或者只有右子树(右斜树).这就是斜树,应用较少 2.满二叉树 所有的分支结点都存在左子树和右子树,并且所有的叶子结点都在同一层上,这样就是满二叉树.就是完美圆满的意思,关键在于树的平衡. 根据满二叉树的定义,得到其特点为: 叶子只能出现

数据结构——二叉树遍历之“递归与非递归遍历”

简述 二叉树的遍历分为先序遍历.中序遍历和后序遍历.如下图所示: 递归遍历 private void bianli1(List<Integer> list, TreeNode root) { // 先序遍历 if (root == null) { return; } list.add(root.val); bianli1(list, root.left); bianli1(list, root.right); } private void bianli2(List<Integer>

数据结构——二叉树概述及其数组(顺序存储)表达法

树与二叉树: 什么是树呢?就是一个节点上会有很多分叉的数据结构.一般的,对于一棵树,我们需要的结构体为一个数据块和几个指针块,这就相当于很多个链表交织在了一起,实际上,链表也可以算是一种特殊的树,而我要讲的,也是一种特殊的树--二叉树. 对于树的各个节点,都有两个属性,称为度(degree),他的意思就是这个节点所拥有的子节点的数量.还有一个属性,称为深度(depth),指节点到根的距离. 什么是二叉树呢?顾名思义,就是度为二的树,它长这样: 如图所示,在链表中我们需要头(head),而在树中我

Java数据结构-二叉树及其遍历

二叉树的定义:n(n>=0)个结点的有限集合,该集合或者为空集(称为空二叉树),或者由一个根结点和两棵互相不相交的.分别称为根结点的左子树和右子树的二叉树组成. 二叉树的特点: 0<=度<=2: 左右子树是有顺序的,不能颠倒: 不论有几棵子树,也要区分它是左子树还是右子树. 二叉树的五种基本形态: 空二叉树: 只有一个根结点: 根结点只有左子树: 根结点只有右子树: 根结点既有左子树又有右子树. 举例3个结点的二叉树的形态有: 下面说一些特殊的二叉树. 斜树:所有的结点都只有左子树的二叉