package avl2
import "fmt"
const LH = 1 /* 左高 */
const EH = 0 /* 等高 */
const RH = -1 /* 右高 */
type AvlTreeNode struct {
data int
bf int /* 结点的平衡因子 */
lchild, rchild *AvlTreeNode
}
type AvlTree struct {
root *AvlTreeNode
}
type Status bool
//参考实现:https://www.cnblogs.com/polly333/p/4798944.html
//对以p为根的二叉排序树作右旋处理
//处理之后p指向新的树根结点,即旋转处理之前的左子树的根结点
//右旋-顺时针旋转(如LL型就得对根结点做该旋转)
func (t *AvlTree) R_Rotate(p *AvlTreeNode) *AvlTreeNode {
L := p.lchild // L指向P的左子树根结点
p.lchild = L.rchild // L的右子树挂接为P的左子树
L.rchild = p
p = L /* P指向新的根结点 */
return p
}
/* 对以P为根的二叉排序树作左旋处理, */
/* 处理之后P指向新的树根结点,即旋转处理之前的右子树的根结点0 */
//左旋-逆时针旋转(如RR型就得对根结点做该旋转)
func (t *AvlTree) L_Rotate(p *AvlTreeNode) *AvlTreeNode {
R := p.rchild // R指向P的右子树根结点
p.rchild = R.lchild // R的左子树挂接为P的右子树
R.lchild = p
p = R // P指向新的根结点
return p
}
/* 对以指针T所指结点为根的二叉树作左平衡旋转处理 */
/* 本算法结束时,指针T指向新的根结点 */
func (t *AvlTree) LeftBalance(T *AvlTreeNode) *AvlTreeNode {
L := T.lchild /* L指向T的左子树根结点 */
switch L.bf {
/* 检查T的左子树的平衡度,并作相应平衡处理 */
case LH: /* 新结点插入在T的左孩子的左子树上,要作单右旋处理 */
T.bf = EH
L.bf = EH
T = t.R_Rotate(T)
break
case RH: /* 新结点插入在T的左孩子的右子树上,要作双旋处理 */ //
Lr := L.rchild /* Lr指向T的左孩子的右子树根 */
switch Lr.bf {
/* 修改T及其左孩子的平衡因子 */
case LH:
T.bf = RH
L.bf = EH
break
case EH:
T.bf = EH
L.bf = EH
break
case RH:
T.bf = EH
L.bf = LH
break
}
Lr.bf = EH
T.lchild = t.L_Rotate(T.lchild) /* 对T的左子树作左旋平衡处理 */
T = t.R_Rotate(T) /* 对T作右旋平衡处理 */
}
return T
}
/* 对以指针T所指结点为根的二叉树作右平衡旋转处理, */
/* 本算法结束时,指针T指向新的根结点 */
func (t *AvlTree) RightBalance(T *AvlTreeNode) *AvlTreeNode {
R := T.rchild /* R指向T的右子树根结点 */
switch R.bf {
/* 检查T的右子树的平衡度,并作相应平衡处理 */
case RH: /* 新结点插入在T的右孩子的右子树上,要作单左旋处理 */
T.bf = EH
R.bf = EH
T = t.L_Rotate(T)
break
case LH: /* 新结点插入在T的右孩子的左子树上,要作双旋处理 */ //最小不平衡树的根结点为负,其右孩子为正
Rl := R.lchild /* Rl指向T的右孩子的左子树根 */
switch Rl.bf {
/* 修改T及其右孩子的平衡因子 */
case RH:
(*T).bf = LH
R.bf = EH
break
case EH:
T.bf = EH
R.bf = EH
break
case LH:
(*T).bf = EH
R.bf = RH
break
}
Rl.bf = EH
T.rchild = t.R_Rotate(T.rchild) /* 对T的右子树作右旋平衡处理 */
T = t.L_Rotate(T) /* 对T作左旋平衡处理 */
}
return T
}
/* 若在平衡的二叉排序树T中不存在和e有相同关键字的结点,则插入一个 */
/* 数据元素为e的新结点,并返回1,否则返回0。若因插入而使二叉排序树 */
/* 失去平衡,则作平衡旋转处理,布尔变量taller反映T长高与否。 */
func (t *AvlTree) InsertAVL(T *AvlTreeNode, data int, taller *Status) (bool, *AvlTreeNode) {
if T == nil {
T = &AvlTreeNode{data, EH, nil, nil}
*taller = true
return true, T
}
if T.data == data {
/* 树中已存在和e有相同关键字的结点则不再插入 */
*taller = false
return false, nil
}
if data < T.data {
flag, P := t.InsertAVL(T.lchild, data, taller)
T.lchild = P
if !flag {
return false, nil
}
if *taller {
/* 已插入到T的左子树中且左子树“长高” */
switch (*T).bf { /* 检查T的平衡度 */
case LH: /* 原本左子树比右子树高,需要作左平衡处理 */
T = t.LeftBalance(T)
*taller = false
break
case EH: /* 原本左、右子树等高,现因左子树增高而使树增高 */
(*T).bf = LH
*taller = true
break
case RH: /* 原本右子树比左子树高,现左、右子树等高 */
(*T).bf = EH
*taller = false
break
}
}
} else {
flag, P := t.InsertAVL(T.rchild, data, taller)
T.rchild = P
if !flag {
return false, nil
}
if *taller {
/* 已插入到T的右子树且右子树“长高” */
switch T.bf { /* 检查T的平衡度 */
case LH: /* 原本左子树比右子树高,现左、右子树等高 */
T.bf = EH
*taller = false
break
case EH: /* 原本左、右子树等高,现因右子树增高而使树增高 */
T.bf = RH
*taller = true
break
case RH: /* 原本右子树比左子树高,需要作右平衡处理 */
T = t.RightBalance(T)
*taller = !true
break
}
}
}
return true, T
}
func InOrderTraverse(t *AvlTreeNode) {
if t != nil {
InOrderTraverse(t.lchild)
fmt.Printf("%d ", t.data)
InOrderTraverse(t.rchild)
}
}
func RunTest() {
a := [...]int{17, 10, 30, 22, 40, 33, 2, 8, 6}
t := &AvlTree{nil}
var taller Status = false
for _, v := range a {
_, P := t.InsertAVL(t.root, v, &taller)
t.root = P
}
InOrderTraverse(t.root)
}
优化指针
package avl2_update
import "fmt"
const LH = 1 /* 左高 */
const EH = 0 /* 等高 */
const RH = -1 /* 右高 */
type AvlTreeNode struct {
data int
bf int /* 结点的平衡因子 */
lchild, rchild *AvlTreeNode
}
type BiTree *AvlTreeNode
type Status bool
func R_Rotate(p *BiTree) {
L := (*p).lchild
(*p).lchild = L.rchild
L.rchild = *p
*p = L
}
func L_Rotate(p *BiTree) {
R := (*p).rchild
(*p).rchild = R.lchild
R.lchild = *p
*p = R
}
func LeftBalance(T *BiTree) {
L := (*T).lchild
switch L.bf {
case LH:
(*T).bf = EH
L.bf = EH
R_Rotate(T)
break
case RH:
Lr := L.rchild
switch Lr.bf {
case LH:
(*T).bf = RH
L.bf = EH
break
case EH:
(*T).bf = EH
L.bf = EH
break
case RH:
(*T).bf = EH
L.bf = LH
break
}
Lr.bf = EH
L_Rotate((*BiTree)(&(*T).lchild))
R_Rotate(T)
}
}
func RightBalance(T *BiTree) {
R := (*T).rchild
switch R.bf {
case RH:
(*T).bf = EH
R.bf = EH
L_Rotate(T)
break
case LH:
Rl := R.lchild
switch Rl.bf {
case RH:
(*T).bf = LH
R.bf = EH
break
case EH:
(*T).bf = EH
R.bf = EH
break
case LH:
(*T).bf = EH
R.bf = RH
break
}
Rl.bf = EH
R_Rotate((*BiTree)(&(*T).rchild))
L_Rotate(T)
}
}
func InsertAVL(T *BiTree, data int, taller *Status) bool {
if *T == nil {
M := &AvlTreeNode{data, EH, nil, nil}
*T = M
*taller = true
return true
}
if (*T).data == data {
*taller = false
return false
}
if data < (*T).data {
flag := InsertAVL((*BiTree)(&(*T).lchild), data, taller)
if !flag {
return false
}
if *taller {
switch (*T).bf {
case LH:
LeftBalance(T)
*taller = false
break
case EH:
(*T).bf = LH
*taller = true
break
case RH:
(*T).bf = EH
*taller = false
break
}
}
} else {
flag := InsertAVL((*BiTree)(&(*T).rchild), data, taller)
if !flag {
return false
}
if *taller {
switch (*T).bf {
case LH:
(*T).bf = EH
*taller = false
break
case EH:
(*T).bf = RH
*taller = true
break
case RH:
RightBalance(T)
*taller = false
break
}
}
}
return true
}
func InOrderTraverse(t *AvlTreeNode) {
if t != nil {
InOrderTraverse(t.lchild)
fmt.Printf("%d ", t.data)
InOrderTraverse(t.rchild)
}
}
func RunTest() {
a := [...]int{3, 2, 1, 7, 9, 6, 4, 5, 8, 10}
var T BiTree = nil
var taller Status = false
for _, v := range a {
InsertAVL(&T, v, &taller)
}
InOrderTraverse(T)
}