[导入]C++实践笔记(四)----AVL树的简单实现
关于AVL树的分析,请见:数据结构与算法分析学习笔记(二)--AVL树的算法思路整理
这里给出AVL树的结构定义以及insert,remove和print三种操作的例程:
1 #include <iostream>
2 #include <list>
3 #include <utility>
4 #include < string>
5 using namespace std;
6
7 static const string PRINT_SPACES = " ";
8
9 template <typename Object>
10 class AVLTree
11 {
12 public:
13 AVLTree(){root=NULL;}
14
15 void insert( const Object &x)
16 {insert(x,root);}
17 void remove( const Object &x)
18 {remove(x,root);}
19
20 void print();
21
22 private:
23 struct AVLnode
24 {
25 Object data;
26 AVLnode *left;
27 AVLnode *right;
28 int height;
29
30 AVLnode(Object ob,AVLnode *l,AVLnode *r, int h= 0):data(ob),left(l),right(r),height(h){}
31 };
32
33 AVLnode *root;
34
35 void insert( const Object &x,AVLnode * &t);
36
37 void remove( const Object &x,AVLnode *&t);
38
39 void leftSingleRotation(AVLnode * &t);
40 void rightSingleRotation(AVLnode * &t);
41
42 void leftDoubleRotation(AVLnode * &t);
43 void rightDoubleRotation(AVLnode * &t);
44
45 int height(AVLnode * &t)
46 {
47 // 结点的高度,空结点的高度为-1
48 return t==NULL?- 1:t->height;
49 }
50
51 int max( int a, int b)
52 {
53 return a<b?b:a;
54 }
55
56 int powerOf2( int x) // 求2的x次方(x大于等于0)
57 {
58 if(x== 0) return 1;
59 int m= 1;
60 while(--x>= 0)
61 m*= 2;
62 return m;
63 }
64
65 AVLnode * max_node(AVLnode * t) // max和min使用递归的话不能使用引用形参,不能返回引用
66 {
67 if(!t)
68 return NULL;
69 if(t->right)
70 return max_node(t->right);
71 else
72 return t;
73 }
74
75 AVLnode * min_node(AVLnode * t)
76 {
77 if(t) // 考虑一下t为空的情况,使其更全面,毕竟这个函数是可以加一个public版本的重载供用户使用
78 while(t->left)
79 t=t->left;
80 return t;
81 }
82 };
83
84 template <typename Object>
85 void AVLTree<Object>::insert( const Object &x,AVLnode * &t)
86 {
87 if(!t)
88 t= new AVLnode(x,NULL,NULL);
89 else if(x<t->data)
90 {
91 insert(x,t->left);
92 if(height(t->left)-height(t->right)== 2) // 在左子树插入结点后,不可能右子树比左子树高2
93 if(x<t->left->data)
94 leftSingleRotation(t); // 左单旋转
95 else
96 leftDoubleRotation(t); // 左双旋转
97 else // 不需要调整就满足平衡条件的话,只需要重新计算其高度就好
98 t->height=max(height(t->left),height(t->right))+ 1;
99 }
100 else if(x>t->data)
101 {
102 insert(x,t->right);
103 if(height(t->right)-height(t->left)== 2)
104 if(x>t->right->data)
105 rightSingleRotation(t); // 右单旋转
106 else
107 rightDoubleRotation(t); // 右双旋转
108 else
109 t->height=max(height(t->left),height(t->right))+ 1;
110 }
111 else; // 不考虑重复的值
112 }
113
114 template <typename Object>
115 void AVLTree<Object>::leftSingleRotation(AVLnode * &t) // 左单旋转
116 {
117 AVLnode *s=t->left;
118 t->left=s->right;
119 s->right=t;
120 t->height=max(height(t->left),height(t->right))+ 1; // 重新计算s和t的高度
121 s->height=max(height(s->left),t->height)+ 1;
122 t=s;
123 }
124
125 template <typename Object>
126 void AVLTree<Object>::rightSingleRotation(AVLnode * &t)
127 {
128 AVLnode *s=t->right;
129 t->right=s->left;
130 s->left=t;
131 t->height=max(height(t->left),height(t->right))+ 1;
132 s->height=max(t->height,height(s->right))+ 1;
133 t=s;
134 }
135
136 template <typename Object>
137 void AVLTree<Object>::leftDoubleRotation(AVLnode * &t) // 左双旋转
138 {
139 AVLnode *p=t->left;
140 AVLnode *q=p->right;
141 t->left=q->right;
142 p->right=q->left;
143 q->left=p;
144 q->right=t;
145 t->height=max(height(t->left),height(t->right))+ 1; // 重新计算3个结点的高度
146 p->height=max(height(p->left),height(p->right))+ 1;
147 q->height=max(p->height,t->height)+ 1;
148 t=q;
149 }
150
151 template <typename Object>
152 void AVLTree<Object>::rightDoubleRotation(AVLnode * &t)
153 {
154 AVLnode *p=t->right;
155 AVLnode *q=p->left;
156 t->right=q->left;
157 p->left=q->right;
158 q->right=p;
159 q->left=t;
160 t->height=max(height(t->left),height(t->right))+ 1;
161 p->height=max(height(p->left),height(p->right))+ 1;
162 q->height=max(t->height,p->height)+ 1;
163 t=q;
164 }
165
166 template <typename Object>
167 void AVLTree<Object>::remove( const Object &x,AVLnode *&t)
168 {
169 if(!t) return; // 没有找到要删除的值,do nothing
170 if(x<t->data)
171 {
172 remove(x,t->left);
173 if(height(t->right)-height(t->left)== 2)
174 {
175 // 右子树比左子树高2,那么在删除操作之前右子树比左子树高1,
176 // 也就是说t的右子树必然不为空,甚至必然有非空子树(高度至少为1).
177 AVLnode *s=t->right;
178 if(height(s->left)>height(s->right))
179 rightDoubleRotation(t); // 右双旋转
180 else
181 rightSingleRotation(t); // 右单旋转
182 }
183 else
184 // 不需要调整就满足平衡条件的话,只需要重新计算其高度就好
185 t->height=max(height(t->left),height(t->right))+ 1;
186 }
187 else if(x>t->data)
188 {
189 remove(x,t->right);
190 if(height(t->left)-height(t->right)== 2)
191 {
192 AVLnode *s=t->left;
193 if(height(s->right)>height(s->left))
194 leftDoubleRotation(t); // 左双旋转
195 else
196 leftSingleRotation(t); // 左单旋转
197 }
198 else
199 t->height=max(height(t->left),height(t->right))+ 1;
200 }
201 else
202 {
203 if(t->left&&t->right)
204 // t的左右子树都非空,把remove操作转移到只有一个非空子树的结点或者叶子结点上去
205 {
206 if(height(t->left)>height(t->right))
207 // 把remove操作往更高的那颗子树上转移
208 {
209 // 左子树中的最大值
210 t->data=max_node(t->left)->data;
211 remove(t->data,t->left);
212 }
213 else
214 {
215 // 右子树中的最小值
216 t->data=min_node(t->right)->data;
217 remove(t->data,t->right);
218 }
219 }
220 else
221 {
222 AVLnode *oldnode=t;
223 t=t->left?t->left:t->right;
224 delete oldnode;
225 }
226 }
227 }
228
229 // 使用队列层次遍历打印树
230 template <typename Object>
231 void AVLTree<Object>::print()
232 {
233 if(!root) return;
234 list<pair<AVLnode *, int> > lis; // int是标志位,0表示正常值,-1表示此处没有值,应打印空格
235 lis.push_back(make_pair(root, 0));
236 AVLnode *em= new AVLnode( 0,NULL,NULL); // 一个空的点
237 int count= 1,j= 0; // count表示当前行的最大结点数,j表示当前行已打印的结点数(空结点也计数)
238 int hg=root->height; // 当前行的高度,计算空格时使用
239
240 while(hg>= 0) // 当hg<0时说明最后一行(树叶)已经打印完毕
241 {
242 while(j++!=count)
243 {
244 for( int i= 1;i<powerOf2(hg);i++)
245 cout<<PRINT_SPACES; // 打印前一部分空格
246 if(lis.front().second== 0)
247 cout<<lis.front().first->data;
248 else
249 cout<<PRINT_SPACES; // int位为-1,则打印空格以对齐
250 if(lis.front().first->left) // 左子树入队
251 lis.push_back(make_pair(lis.front().first->left, 0));
252 else
253 lis.push_back(make_pair(em,- 1));
254 if(lis.front().first->right) // 右子树入队
255 lis.push_back(make_pair(lis.front().first->right, 0));
256 else
257 lis.push_back(make_pair(em,- 1));
258 for( int i= 0;i<powerOf2(hg);i++)
259 cout<<PRINT_SPACES; // 打印后一部分空格
260 lis.pop_front();
261 }
262 j= 0;
263 count*= 2; // 下一行的最大节点数是上一行的两倍
264 --hg; // 高度减1
265 cout<<endl; // 换行
266 }
267 delete em;
268 lis.clear();
269 }
2 #include <list>
3 #include <utility>
4 #include < string>
5 using namespace std;
6
7 static const string PRINT_SPACES = " ";
8
9 template <typename Object>
10 class AVLTree
11 {
12 public:
13 AVLTree(){root=NULL;}
14
15 void insert( const Object &x)
16 {insert(x,root);}
17 void remove( const Object &x)
18 {remove(x,root);}
19
20 void print();
21
22 private:
23 struct AVLnode
24 {
25 Object data;
26 AVLnode *left;
27 AVLnode *right;
28 int height;
29
30 AVLnode(Object ob,AVLnode *l,AVLnode *r, int h= 0):data(ob),left(l),right(r),height(h){}
31 };
32
33 AVLnode *root;
34
35 void insert( const Object &x,AVLnode * &t);
36
37 void remove( const Object &x,AVLnode *&t);
38
39 void leftSingleRotation(AVLnode * &t);
40 void rightSingleRotation(AVLnode * &t);
41
42 void leftDoubleRotation(AVLnode * &t);
43 void rightDoubleRotation(AVLnode * &t);
44
45 int height(AVLnode * &t)
46 {
47 // 结点的高度,空结点的高度为-1
48 return t==NULL?- 1:t->height;
49 }
50
51 int max( int a, int b)
52 {
53 return a<b?b:a;
54 }
55
56 int powerOf2( int x) // 求2的x次方(x大于等于0)
57 {
58 if(x== 0) return 1;
59 int m= 1;
60 while(--x>= 0)
61 m*= 2;
62 return m;
63 }
64
65 AVLnode * max_node(AVLnode * t) // max和min使用递归的话不能使用引用形参,不能返回引用
66 {
67 if(!t)
68 return NULL;
69 if(t->right)
70 return max_node(t->right);
71 else
72 return t;
73 }
74
75 AVLnode * min_node(AVLnode * t)
76 {
77 if(t) // 考虑一下t为空的情况,使其更全面,毕竟这个函数是可以加一个public版本的重载供用户使用
78 while(t->left)
79 t=t->left;
80 return t;
81 }
82 };
83
84 template <typename Object>
85 void AVLTree<Object>::insert( const Object &x,AVLnode * &t)
86 {
87 if(!t)
88 t= new AVLnode(x,NULL,NULL);
89 else if(x<t->data)
90 {
91 insert(x,t->left);
92 if(height(t->left)-height(t->right)== 2) // 在左子树插入结点后,不可能右子树比左子树高2
93 if(x<t->left->data)
94 leftSingleRotation(t); // 左单旋转
95 else
96 leftDoubleRotation(t); // 左双旋转
97 else // 不需要调整就满足平衡条件的话,只需要重新计算其高度就好
98 t->height=max(height(t->left),height(t->right))+ 1;
99 }
100 else if(x>t->data)
101 {
102 insert(x,t->right);
103 if(height(t->right)-height(t->left)== 2)
104 if(x>t->right->data)
105 rightSingleRotation(t); // 右单旋转
106 else
107 rightDoubleRotation(t); // 右双旋转
108 else
109 t->height=max(height(t->left),height(t->right))+ 1;
110 }
111 else; // 不考虑重复的值
112 }
113
114 template <typename Object>
115 void AVLTree<Object>::leftSingleRotation(AVLnode * &t) // 左单旋转
116 {
117 AVLnode *s=t->left;
118 t->left=s->right;
119 s->right=t;
120 t->height=max(height(t->left),height(t->right))+ 1; // 重新计算s和t的高度
121 s->height=max(height(s->left),t->height)+ 1;
122 t=s;
123 }
124
125 template <typename Object>
126 void AVLTree<Object>::rightSingleRotation(AVLnode * &t)
127 {
128 AVLnode *s=t->right;
129 t->right=s->left;
130 s->left=t;
131 t->height=max(height(t->left),height(t->right))+ 1;
132 s->height=max(t->height,height(s->right))+ 1;
133 t=s;
134 }
135
136 template <typename Object>
137 void AVLTree<Object>::leftDoubleRotation(AVLnode * &t) // 左双旋转
138 {
139 AVLnode *p=t->left;
140 AVLnode *q=p->right;
141 t->left=q->right;
142 p->right=q->left;
143 q->left=p;
144 q->right=t;
145 t->height=max(height(t->left),height(t->right))+ 1; // 重新计算3个结点的高度
146 p->height=max(height(p->left),height(p->right))+ 1;
147 q->height=max(p->height,t->height)+ 1;
148 t=q;
149 }
150
151 template <typename Object>
152 void AVLTree<Object>::rightDoubleRotation(AVLnode * &t)
153 {
154 AVLnode *p=t->right;
155 AVLnode *q=p->left;
156 t->right=q->left;
157 p->left=q->right;
158 q->right=p;
159 q->left=t;
160 t->height=max(height(t->left),height(t->right))+ 1;
161 p->height=max(height(p->left),height(p->right))+ 1;
162 q->height=max(t->height,p->height)+ 1;
163 t=q;
164 }
165
166 template <typename Object>
167 void AVLTree<Object>::remove( const Object &x,AVLnode *&t)
168 {
169 if(!t) return; // 没有找到要删除的值,do nothing
170 if(x<t->data)
171 {
172 remove(x,t->left);
173 if(height(t->right)-height(t->left)== 2)
174 {
175 // 右子树比左子树高2,那么在删除操作之前右子树比左子树高1,
176 // 也就是说t的右子树必然不为空,甚至必然有非空子树(高度至少为1).
177 AVLnode *s=t->right;
178 if(height(s->left)>height(s->right))
179 rightDoubleRotation(t); // 右双旋转
180 else
181 rightSingleRotation(t); // 右单旋转
182 }
183 else
184 // 不需要调整就满足平衡条件的话,只需要重新计算其高度就好
185 t->height=max(height(t->left),height(t->right))+ 1;
186 }
187 else if(x>t->data)
188 {
189 remove(x,t->right);
190 if(height(t->left)-height(t->right)== 2)
191 {
192 AVLnode *s=t->left;
193 if(height(s->right)>height(s->left))
194 leftDoubleRotation(t); // 左双旋转
195 else
196 leftSingleRotation(t); // 左单旋转
197 }
198 else
199 t->height=max(height(t->left),height(t->right))+ 1;
200 }
201 else
202 {
203 if(t->left&&t->right)
204 // t的左右子树都非空,把remove操作转移到只有一个非空子树的结点或者叶子结点上去
205 {
206 if(height(t->left)>height(t->right))
207 // 把remove操作往更高的那颗子树上转移
208 {
209 // 左子树中的最大值
210 t->data=max_node(t->left)->data;
211 remove(t->data,t->left);
212 }
213 else
214 {
215 // 右子树中的最小值
216 t->data=min_node(t->right)->data;
217 remove(t->data,t->right);
218 }
219 }
220 else
221 {
222 AVLnode *oldnode=t;
223 t=t->left?t->left:t->right;
224 delete oldnode;
225 }
226 }
227 }
228
229 // 使用队列层次遍历打印树
230 template <typename Object>
231 void AVLTree<Object>::print()
232 {
233 if(!root) return;
234 list<pair<AVLnode *, int> > lis; // int是标志位,0表示正常值,-1表示此处没有值,应打印空格
235 lis.push_back(make_pair(root, 0));
236 AVLnode *em= new AVLnode( 0,NULL,NULL); // 一个空的点
237 int count= 1,j= 0; // count表示当前行的最大结点数,j表示当前行已打印的结点数(空结点也计数)
238 int hg=root->height; // 当前行的高度,计算空格时使用
239
240 while(hg>= 0) // 当hg<0时说明最后一行(树叶)已经打印完毕
241 {
242 while(j++!=count)
243 {
244 for( int i= 1;i<powerOf2(hg);i++)
245 cout<<PRINT_SPACES; // 打印前一部分空格
246 if(lis.front().second== 0)
247 cout<<lis.front().first->data;
248 else
249 cout<<PRINT_SPACES; // int位为-1,则打印空格以对齐
250 if(lis.front().first->left) // 左子树入队
251 lis.push_back(make_pair(lis.front().first->left, 0));
252 else
253 lis.push_back(make_pair(em,- 1));
254 if(lis.front().first->right) // 右子树入队
255 lis.push_back(make_pair(lis.front().first->right, 0));
256 else
257 lis.push_back(make_pair(em,- 1));
258 for( int i= 0;i<powerOf2(hg);i++)
259 cout<<PRINT_SPACES; // 打印后一部分空格
260 lis.pop_front();
261 }
262 j= 0;
263 count*= 2; // 下一行的最大节点数是上一行的两倍
264 --hg; // 高度减1
265 cout<<endl; // 换行
266 }
267 delete em;
268 lis.clear();
269 }
作者: Barryhe 发表于 2011-11-28 16:30 原文链接
评论: 0 查看评论 发表评论
最新新闻:
· Google雅虎等呼吁欧盟勿制订过于严格隐私法律(2011-11-28 21:06)
· 强龙难压地头蛇 微软WP7俄罗斯忍痛割Bing(2011-11-28 20:27)
· 奇艺启动新域名背后:解决视听许可证遗留问题(2011-11-28 20:25)
· Kindle“黑色星期五”销量达去年同期四倍(2011-11-28 20:24)
· 维基解密本周将启动新在线系统(2011-11-28 20:21)
编辑推荐:如何快速成为Javascript高手的思考
网站导航:博客园首页 我的园子 新闻 闪存 博问 知识库
文章来源: http://www.cnblogs.com/heqile/archive/2011/11/28/2266385.html