java数据结构和算法(一):二叉树

  有序数组的查找速度很快,但是插入一个数据项时,就必须先找到插入数据项的位置,然后将后面所有的数据项后移一位,平均要移动N/2次,这是很费时的,删除数据也是。

  链表的插入和删除很快,只需要改变引用值就行了,但是查找数据却很慢,需要从链表的第一个节点查找到所需要的数据项为止,平均需要比较N/2次。

  树这种数据结构就具有了两者的优点,数组查找快以及链表插入、删除快。

1.树的定义:

  树是n个有限个数据点通过连接他们的边组成的具有层次关系的集合。

  节点高度:从叶节点向上数,连接该节点的边数就是节点的高度

  节点深度:从根节点往下数,该节点所在的层数就是节点的深度。

  树中节点高度和深度的最大值就是树的高度和深度。注意:有两种说法,一种是根的深度为0,一种根的深度为1,所以树的深度取决于根节点所在的层数是0还是1。

2.二叉树的性质:

  (1)两种特殊的树:满二叉树和完全二叉树。在一棵二叉树中,如果所有分支节点都存在左子树和右子树,并且所有的叶节点都在同一层上称为满二叉树;对于一颗具有n个节点的二叉树按层序编号,如果编号为i(0<i<=n)的节点与同样深度的满二叉树中编号为i的节点在二叉树中位置完全相同,则称完全二叉树。

  (2)在二叉树中的第i层上至多有2^(i-1)个节点(i>=1)

  (3)深度为k的二叉树至多有2^k-1个节点(k>=1)

    (4)  对于任何一颗二叉树T,如果其叶子节点数为n0,度为2的节点数为n2,则n0 = n2+1

扫描二维码关注公众号,回复: 10668108 查看本文章

  (5)具有n个节点的完全二叉树的深度为|log2n|+1(|log2n|为不大于它自身的最大整数)。

3.二叉树的插入、查找和删除

  这里主要讲二叉搜索树的操作。首先谈二叉搜索树的简单定义:如果节点不为空,有右子节点》根节点》左子节点,满足这样条件的树称为二叉搜索树。

插入:

public void insert(int data){
if(root==null){
root = new Node(data);
return;
}
Node temp = root;
Node pre = new Node();//获取待插入节点的父节点
while(temp!=null){
pre = temp;
if(temp.data==data){
return;
}else if(temp.data>data){
temp = temp.leftnode;
}else{
temp = temp.rightnode;
}
}
if(pre.data>data){
pre.leftnode = new Node(data);
}else{
pre.rightnode = new Node(data);
}

}

查找:

public Node search(int data){
Node temp = root;
while(temp!=null){
if(temp.data==data){
return temp;
}else if(temp.data>data){
temp = temp.leftnode;
}else{
temp = temp.rightnode;
}
}
return new Node();
}

删除:

分情况讨论: 

 (1)删除的节点如果没有右节点则直接用左子节点替换。

 (2)否则,如果删除节点的右节点不存在左子树,则直接用该右子节点替换。

 (3)否则,找到右节点的左子树中最左的节点替换被删除的节点,然后将替换节点的右子节点放到替换节点的位置。

 (1)第一种情况         (2)第二种情况

 (3)第三种情况

 1 public boolean delete(int data){
 2         Node temp = root;
 3         Node pre = new Node();
 4         pre.leftnode = root;
 5         byte state = 0;               // 0代表左子节点,1代表右子节点
 6         while(temp!=null&&temp.data!=data){    //找到待删除的节点
 7             pre = temp;
 8             if(temp.data>data){
 9                 temp = temp.leftnode;
10                 state = 0;
11             }else{
12                 temp = temp.rightnode;
13                 state = 1;
14             }
15         }
16         if(temp==null) return false;
17 
18         Node templ = temp.leftnode;      //保存待删除节点的左子树和右子树
19         Node tempr = temp.rightnode;
20 
21         if(tempr==null){            //第一种情况
22             connectNode(pre,templ,state);
23         }else if(tempr.leftnode==null){   //第二种情况
24             connectNode(pre,tempr,state);
25             tempr.leftnode = templ;
26         }else{                  //第三种情况
27             Node pre2 = tempr;
28             temp = pre2.leftnode;
29             while(temp.leftnode!=null){
30                 pre2 = temp;
31                 temp = temp.leftnode;
32             }
33             pre2.leftnode = temp.rightnode;
34 
35             temp.leftnode = templ;
36             temp.rightnode = tempr;
37             if(pre.leftnode==root) root = temp;
38             connectNode(pre,temp,state);
39         }
40         return true;
41     }

private void connectNode(Node pre, Node node, byte state) {
    if(state==0){
pre.leftnode = node;
}else{
pre.rightnode = node;
}
}

  通过上面的删除分类讨论,我们发现删除其实是挺复杂的,那么其实我们可以不用真正的删除该节点,只需要在Node类中增加一个标识字段isDelete,当该字段为true时,表示该节点已经删除,反正没有删除。那么我们在做比如find()等操作的时候,要先判断isDelete字段是否为true。这样删除的节点并不会改变树的结构。

4.遍历二叉树

(1)前序遍历:父节点=》左子节点=》右子节点

递归算法:

1 public void beforeOrder(Node node){
2         if(node==null){
3             return;
4         }
5         System.out.println(node.data);
6         beforeOrder(node.leftnode);
7         beforeOrder(node.rightnode);
8     }

非递归算法:

 1     public void beforeOrder2(Node node){
 2         Stack<Node> stack = new Stack<>();
 3         stack.push(node);
 4         while(!stack.isEmpty()){
 5             Node pop = stack.pop();
 6             if (pop==null) continue;
 7             System.out.println(pop.data);
 8             stack.push(pop.rightnode);
 9             stack.push(pop.leftnode);
10         }
11 
12     }

(2)中序遍历:左子节点=》父节点=》右子节点

递归算法:

1 public void inOrder(Node node){
2     if(node==null){
3         return;
4     }
5     inOrder(node.leftnode);
6     System.out.println(node.data);
7     inOrder(node.rightnode);
8 }

非递归算法:

 1 public void inOrder2(Node node){
 2     Stack<Node> stack = new Stack<>();
 3     Node cur = node;
 4     while(!stack.isEmpty()||cur!=null){
 5         while(cur!=null){
 6             stack.push(cur);
 7             cur = cur.leftnode;
 8         }
 9         Node pop = stack.pop();
10         System.out.println(pop.data);
11         if(pop.rightnode==null) continue;
12         cur = pop.rightnode;
13     }
14 }

(3)后序遍历:左子节点=》右字节点=》父节点

递归算法:

1 public void afterOrder(Node node){
2     if(node==null){
3         return;
4     }
5     afterOrder(node.leftnode);
6     afterOrder(node.rightnode);
7     System.out.println(node.data);
8 }

非递归算法:

 1 public void afterOrder2(Node node){
 2     Stack<Node> stack = new Stack<>();
 3     stack.push(node);
 4     List<Node> mylist = new ArrayList<>();
 5     while(!stack.isEmpty()){
 6         Node pop = stack.pop();
 7         if (pop==null) continue;
 8         mylist.add(pop);
 9         stack.push(pop.leftnode);
10         stack.push(pop.rightnode);
11     }
12     Collections.reverse(mylist);
13     for (Node node1 : mylist) {
14         System.out.println(node1.data);
15     }
16 }

(4)层次遍历

 1 public void levelOrder(Node node){
 2     Queue<Node> queue = new LinkedList<>();
 3     queue.offer(node);
 4     while(!queue.isEmpty()){
 5         int size = queue.size();
 6         while(size-->0){
 7             Node node1 = queue.poll();
 8             if(node1==null){
 9                 continue;
10             }
11             System.out.println(node1.data);
12             queue.offer(node1.leftnode);
13             queue.offer(node1.rightnode);
14         }
15     }
16 }

5.效率分析

  在插入的数据是无序的情况下,插入的效率是logN;在二叉树是平衡树的情况下,删除、查找的效率都为logN;但是树的遍历效率比较低。

猜你喜欢

转载自www.cnblogs.com/amazing-eight/p/12672054.html
今日推荐