算法导论 第十二章 二叉搜索树(python)
上图:
这是二叉搜索树(也有说是查找树的)基本结构:如果y是x的左子树中的一个结点,那么y.key <= x.key(如a图中的6根结点大于它左子树的每一个结点 6 >= {2,5,5}),如果y是x的右子树中的一个结点,那么y.key >x.key
注:不同堆,堆是中间的结点最大或最小,而二叉搜索树是左中右的大小顺序,我们用这个特性来遍历二叉搜索树得到是他的顺序排列(中序遍历)#中在什么地方就叫什么遍历 如前序遍历:中左右 后序:左右中
如图a他的中序遍历为 2->5->5->6->7->8 #从大到小
基本操作:
SEARCH:查找关键字为k的结点 O(h) #h为二叉树的高度
MINIMUM:查找二叉树的最小值(显然是最左的那个结点) O(h)
MAXIMUM:查找二叉树的最大值(显然是最右的那个结点) O(h)
PREDECESSOR:查找x的前驱y O(h)
SUCCESSOR:查找x的后驱y O(h)
INSERT:插入结点z O(h)
DELETE:删除结点z O(h)
注:
1.其中插入和删除因为要调整树的结构所以有点复杂
2.复杂都为O(h) #h为二叉树的高度,我们看下图
他们的结点数(n)都是6但是高度是不同的 ha = 2 而 hb = 4 ,这种差距在应用中可能会有很大的性能问题,同以前的快速排序一样使用随机化方法或者是其他的限定条件(如红黑树)来保证性能在一个好一点的范围内(h = logn),就是不能让二叉树保持一条直线向下
给出python实现:
class Node: #结点 def __init__(self,data):
self.left = None
self.right = None
self.parent = None
self.data = data
def createNode(self,data):
#初始化
return Node(data) def inorder_tree_walk(self):
"""
中序遍历
"""
if self.left: #左
self.left.inorder_tree_walk()
print(self.data,end = ' ') #中
if self.right: #右
self.right.inorder_tree_walk()
def tree_search(self,data):
if self == None or self.data == data:
return self if data < self.data:
return self.left.tree_seatch(data)
else :
return self.right.tree_seatch(data) def iterative_tree_search(self,data):
#非递归版查找一般会比递归版更快
n = self
while n != None and data != n.data:
if data < n.data:
n = n.left
else:
n = n.right
return n def tree_mininum(self):
if self.left:
return self.left.tree_mininum()
else:
return self def tree_maximun(self):
if self.right:
return self.right.true_maximun()
else:
return self def tree_successor(self): """
找后继:
有右子树,取右子树中最小的
没有右子树,也就是这个子树中最大的,应该向上找第一个把他当右子树的结点
前驱 相反
"""
x = self
if x.right != None:
return x.right.tree_mininum()
else:
p = x.parent
while p and p.right == x:
x = p
p = p.parent
return p
def tree_predecessor(self):
x = self
if x.left != None:
return x.left.tree_maximun()
else:
p = x.parent
while p and p.left == x:
x = p
p = p.parent
return p def tree_insert(self,data):
#插入data
node = self
while node:
if data < node.data:
next = node.left
else:
next = node.right
if next:
node = next
else:
break
nn = self.createNode(data)
if data < node.data:
node.left = nn
node.left.parent = node
else:
node.right = nn
node.right.parent = node
return nn def tree_delete(self,root):
'''
1.有2子树
1.相对树结构移除的是self 2.少于有2子树
1.相对树结点移除的是self的后继 当然删除还是self.data
''' n = self if n.left == None or n.right == None :
y = n #至多有1子树直接删除
else:
y = n.tree_successor()#要用后继覆盖他 if y.left == None: #x 覆盖 y #至多有1个子树 y是黑的话 x的黑+1
x = y.right
else:
x = y.left x.parent = y.parent
if y.parent == None:
root = x
elif y == y.parent.left:
y.parent.left = x
else:
y.parent.right = x if y != n:
n.data = y.data
return root; if __name__ == '__main__':
root = Node(6)
root.tree_insert(5)
root.tree_insert(7)
root.tree_insert(2)
root.tree_insert(3)
root.tree_insert(8)
print("中序遍历: ",end = '')
root.inorder_tree_walk()
print("\n查找:",end = '')
test1= root.iterative_tree_search(5)
print(str(test1.data)+'的后继:'+str(test1.tree_successor().data)) print("查找:",end = '')
test2= root.tree_search(6)
print(str(test2.data)+'的前驱:'+str(test2.tree_predecessor().data))
root = test2.tree_delete(root)
print("删除6后:",end = '')
root.inorder_tree_walk()
root = root.iterative_tree_search(2).tree_delete(root)
print("\n删除2后:",end = '')
root.inorder_tree_walk()
'''
================= RESTART: F:\python\algorithms\12_1_tree.py =================
中序遍历: 2 3 5 6 7 8
查找:5的后继:6
查找:6的前驱:5
删除6后:2 3 5 7 8
删除2后:3 5 7 8
>>> python 3.5.1 win7
'''
参考引用:
http://www.wutianqi.com/?cat=515&paged=4
http://blog.csdn.net/fxjtoday/article/details/6448083
算法导论 第十二章 二叉搜索树(python)的更多相关文章
- 算法导论(Introduction to Algorithms )— 第十二章 二叉搜索树— 12.1 什么是二叉搜索树
搜索树数据结构支持很多动态集合操作,如search(查找).minmum(最小元素).maxmum(最大元素).predecessor(前驱).successor(后继).insert(插入).del ...
- 算法导论第十八章 B树
一.高级数据结构 本章以后到第21章(并查集)隶属于高级数据结构的内容.前面还留了两章:贪心算法和摊还分析,打算后面再来补充.之前的章节讨论的支持动态数据集上的操作,如查找.插入.删除等都是基于简单的 ...
- 【Coding算法导论】第4章:最大子数组问题
Coding算法导论 本系列文章主要针对算法导论一书上的算法,将书中的伪代码用C++实现 代码未经过大量数据测试,如有问题,希望能在回复中指出! (一)问题描述 给定一个数组,求数组中连续的子数组的和 ...
- 小白专场-是否同一颗二叉搜索树-python语言实现
目录 一.二叉搜索树的相同判断 二.问题引入 三.举例分析 四.方法探讨 4.1 中序遍历 4.2 层序遍历 4.3 先序遍历 4.4 后序遍历 五.总结 六.代码实现 一.二叉搜索树的相同判断 二叉 ...
- Leetcode 98 验证二叉搜索树 Python实现
给定一个二叉树,判断其是否是一个有效的二叉搜索树. 假设一个二叉搜索树具有如下特征: 节点的左子树只包含小于当前节点的数. 节点的右子树只包含大于当前节点的数. 所有左子树和右子树自身必须也是二叉搜索 ...
- LeetCode 98. 验证二叉搜索树 | Python
98. 验证二叉搜索树 题目来源:https://leetcode-cn.com/problems/validate-binary-search-tree 题目 给定一个二叉树,判断其是否是一个有效的 ...
- 【原创】《算法导论》链表一章带星习题试解——附C语言实现
原题: 双向链表中,需要三个基本数据,一个携带具体数据,一个携带指向上一环节的prev指针,一个携带指向下一环节的next指针.请改写双向链表,仅用一个指针np实现双向链表的功能.定义np为next ...
- 二叉搜索树(python)
# -*- coding: utf-8 -*- class BSTNode(object): def __init__(self, key, value, left=None, right=None) ...
- LeetCode--096--不同的二叉搜索树(python)
我的思路比较low直接看官方题解吧... class Solution: def numTrees(self, n: int) -> int: G = [0] * (n+1) G[0],G[1] ...
随机推荐
- c++ 常用的几种重载操作符
运算符可以作为普通函数,朋友函数或成员函数来重载.下面的经验法则可以帮助您确定哪种形式最适合于给定的情况: 如果你重载了赋值(=),下标([]),函数调用(())或成员选择( - >),那么它就 ...
- jmeter(十九)调试工具Debug Sampler
一.Debug Sampler介绍: 使用Jmeter开发脚本时,难免需要调试,这时可以使用Jmeter的Debug Sampler,它有三个选项:JMeter properties,JMeter v ...
- Nginx缓存[proxy cache、memcache]
nginx自带缓存 nginx自己有单独的进程来对磁盘上的缓存文件进行扫描,在内存中建立缓存索引.并且有管理进程来对缓存进行过期判断,更新等操作 定义:只能在http段中使用 proxy_cache_ ...
- php Try Catch多层级异常测试
<?php class a { public function a1 () { try { throw new Exception('123'); } catch (Exception $e) ...
- P1936 水晶灯火灵
题目描述 ,刚好符合以下③条规则:①m.n∈{1,2,……,k}②(n²-m*n-m²)²=1③m.n为整数. 输入输出格式 输入格式: Only one:k.(What?还是失败?) 输出格式: 共 ...
- 对js 面对对象编程的一些简单的理解
由简单开始深入: 最简单的 直接对象开始 var desen = { age:24, name:'xyf', job:'fontEnd', getName:function(){ console.lo ...
- os模块详解2
1.os.getenv('HOME') 读取操作系统环境变量HOME的值. 2.os.environ 返回操作系统所有的环境变量. 3.os.environ.setdefault(‘a’,‘b’) ...
- AI学习一:环境安装
对于Python开发用户来讲,PIP安装软件包是家常便饭.但国外的源下载速度实在太慢,浪费时间.而且经常出现下载后安装出错问题.所以把PIP安装源替换成国内镜像,可以大幅提升下载速度,还可以提高安装成 ...
- 回顾PMP考试
2014年9月20日,于我来说绝对可以说是一个重要的日子.经过考场里4个多小时(4个小时正式的时间+前面的签到以及后面的survey等)的鏖战,出去之后才发现北京外国语大学的楼宇是如此的漂亮,阳光也是 ...
- js图片轮播效果常见的产品无缝轮播
<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...