[关闭]
@Scrazy 2017-04-13T08:23:49.000000Z 字数 4998 阅读 1041

二叉树

python

更新 2017-03-22


畏惧了好久的二叉树,终于在近两天开搞了。二分法查找已在前几天完成,磨刀霍霍向猪羊,吼吼吼!

首先,建立一棵树(二叉搜索树)。

  1. class Node:
  2. def __init__(self, data):
  3. self.left = None
  4. self.right = None
  5. self.data = data

这样,光秃秃的小树苗就种好了。接着就是茁长生长啦。浇水去喽!

  1. class Node
  2. '''
  3. ...
  4. '''
  5. def insert(self, data):
  6. if data < self.data: # 树叉小于节点
  7. if self.left is None: # 并且左面的树叉为空
  8. self.left = Node(data) # 当仁不让的插入
  9. else: # 非空的话
  10. self.left.insert(data) # 以左树叉为节点继续插入
  11. elif data > self.data:
  12. if self.right is None:
  13. self.right = Node(data)
  14. else:
  15. self.right.insert(data)
  16. else:
  17. self.data = data

浇完水后,小树苗噌噌的往上窜啊。

  1. class Node
  2. '''
  3. 省略上述代码
  4. '''
  5. def search(self, data, parent=None):
  6. '''
  7. data为目标查询值,同时返回parent(父节点)便于定位。
  8. '''
  9. if data < self.data:
  10. if self.left is None:
  11. return None, None
  12. else:
  13. return self.left.search(data, self)
  14. elif data > self.data:
  15. if self.right is None:
  16. return None, None
  17. return self.right.search(data, self)
  18. else:
  19. # return self.data, parent.data
  20. return self, parent

树苗生长的那么好,想看看每个叉上都是啥呀,来来来,抬头往上看((其实是往下看啦)。

  1. def print_tree(self):
  2. if self.left:
  3. self.left.print_tree()
  4. print(self.data, end=' ')
  5. if self.right:
  6. self.right.print_tree()

树的遍历又分为以下三种:

  1. 前序(root -> left -> right)
  2. 中序(left -> root -> right)
  3. 后序(left -> right -> root)

调整print_tree函数里 print(self.data, end=' ') 的顺序即可实现三种遍历方式。

转眼间小树苗涨的太旺盛了,疯涨啊!!怎么办呢,剪几个枝吧。别怪我哦,小树苗! 删除节点时,有三种可能的情况:

  1. 目标节点下没有任何节点(0个)
  2. 目标节点下有一个节点
  3. 目标节点下有两个节点

判断节点数目程序如下:

  1. class Node
  2. '''
  3. 省略代码
  4. '''
  5. def chrildren(self):
  6. count = 0
  7. if self.left:
  8. count += 1
  9. if self.right:
  10. count += 1
  11. return count

接下来就是删除操作啦。哦吼吼。

  1. class Node
  2. '''
  3. 省略
  4. '''
  5. def delete(self, data):
  6. node, parent = self.search(data)
  7. chrildren = node.chrildren() # 子节点数目
  8. if chrildren == 0: # 情况 1, 没有子节点,直接删除即可
  9. if parent.left is node: # 判断目标节点是其父节点的 左or右 节点
  10. parent.left = None
  11. else:
  12. parent.right = None
  13. del node
  14. elif chrildren == 1: # 情况 2, 有一个子节点,用子节点替换其即可
  15. if node.left:
  16. tmp = node.left
  17. else:
  18. tmp = node.right
  19. if parent:
  20. if parent.left is node:
  21. parent.left = tmp
  22. else:
  23. parent.right = tmp
  24. del node
  25. else:
  26. '''
  27. 第三种情况比较复杂:
  28. 1\. 左节点0个子节点
  29. 2\. 左节点1个子节点
  30. 3\. 左节点2个子节点
  31. '''
  32. parent = node
  33. successor = node.right
  34. while successor.left: # 递归思想,直至找到'最左'的子节点, 保持树的平衡,用右子节点的值替换
  35. parent = successor
  36. successor = successor.left
  37. node.data = successor.data
  38. if parent.left == successor:
  39. parent.left = successor.right
  40. else:
  41. parent.right = successor.right
  42. # 接下来可以测试以下种的树怎么样啦。
  43. root = Node(11) root.insert(14) root.insert(9) root.insert(9) root.insert(7) root.insert(10) root.insert(4) root.insert(5) root.inser)

把自己理解的部分写了写。当做练习,就先当个α版吧。
2016-05-28

基本搞明白了
完整代码在这里

下面是关于二叉树的一些算法

广度遍历和深度遍历二叉树!

  1. def lookup(root):
  2. stack = [root]
  3. while stack:
  4. current = stack.pop()
  5. print(current.data, end=' ')
  6. if current.left:
  7. stack.append(current.left)
  8. if current.right:
  9. stack.append(current.right)
  10. def deep(root):
  11. if not root:
  12. return
  13. deep(root.left)
  14. deep(root.right)
  15. print(root.data, end=' ')

求最大树深

  1. # -*- coding:utf-8 -*-
  2. class TreeNode:
  3. def __init__(self, x):
  4. self.val = x
  5. self.left = None
  6. self.right = None
  7. class Solution:
  8. def TreeDepth(self, pRoot):
  9. if not pRoot:
  10. return 0
  11. return max(self.TreeDepth(pRoot.left), self.TreeDepth(pRoot.right)) + 1

比较两棵树是否相同

  1. def is_same(t1, t2):
  2. if t1 == None and t2 == None:
  3. return True
  4. elif t1 and t2:
  5. return t1.data == t2.data and is_same(t1.left, t2.left)\
  6. and is_same(t1.right, t2.right)
  7. else:
  8. return False

已知前序中序求后序

前面说到:
前序: root -> left -> right
中序: left -> root -> right
后序: left -> right -> root

前序: 第一个值 A 即为根节点
中序: A 的左边全为左子树,右边全是右子树

  1. def pre_in_post(pre_order, in_order):
  2. if not pre_order:
  3. return
  4. post = Node(pre_order[0])
  5. index = in_order.index(pre_order[0])
  6. post.left = pre_in_post(pre_order[1:index+1], in_order[:index])
  7. post.right = pre_in_post(pre_order[index+1:], in_order[index+1:])
  8. return post

已知前序中序构造出树

  1. # -*- coding:utf-8 -*-
  2. class TreeNode:
  3. def __init__(self, x):
  4. self.val = x
  5. self.left = None
  6. self.right = None
  7. class Solution:
  8. # 返回构造的TreeNode根节点
  9. def reConstructBinaryTree(self, pre, tin):
  10. # write code here
  11. if not pre:
  12. return
  13. tree = TreeNode(pre[0])
  14. index = tin.index(pre[0])
  15. tree.left = self.reConstructBinaryTree(pre[1:index+1],tin[:index])
  16. tree.right = self.reConstructBinaryTree(pre[index+1:],tin[index+1:])
  17. return tree
  18. @classmethod
  19. def print_tree(cls, tree):
  20. if tree:
  21. cls.print_tree(tree.left)
  22. cls.print_tree(tree.right)
  23. print(tree.val)
  24. pre = [1,2,3,4,5,6,7]
  25. tin = [3,2,4,1,6,5,7]
  26. s = Solution()
  27. t = s.reConstructBinaryTree(pre, tin)
  28. s.print_tree(t)

树的子结构

  1. pRoot2 的子树是否为 pRoot2
  2. # -*- coding:utf-8 -*-
  3. # class TreeNode:
  4. # def __init__(self, x):
  5. # self.val = x
  6. # self.left = None
  7. # self.right = None
  8. class Solution:
  9. def is_subtree(self, t1, t2):
  10. if not t2: # t2 is None 其为子树
  11. return True
  12. if not t1:
  13. return False
  14. if not t1.val == t2.val:
  15. return False
  16. return self.is_subtree(t1.left, t2.left) and self.is_subtree(t1.right, t2.right)
  17. def HasSubtree(self, pRoot1, pRoot2):
  18. # write code here
  19. result = False
  20. if pRoot1 and pRoot2:
  21. if pRoot1.val == pRoot2.val:
  22. result = self.is_subtree(pRoot1, pRoot2)
  23. if not result:
  24. result = self.is_subtree(pRoot1.left, pRoot2)
  25. if not result:
  26. result = self.is_subtree(pRoot1.right, pRoot2)
  27. return result

对称二叉树

  1. # -*- coding:utf-8 -*-
  2. # class TreeNode:
  3. # def __init__(self, x):
  4. # self.val = x
  5. # self.left = None
  6. # self.right = None
  7. class Solution:
  8. def isSymmetrical(self, pRoot):
  9. def is_same(p1, p2):
  10. if not (p1 or p2):
  11. return True
  12. elif p1 and p2 and p1.val == p2.val:
  13. return is_same(p1.left, p2.right) and is_same(p1.right, p2.left)
  14. return False
  15. if not pRoot:
  16. return True
  17. return is_same(pRoot.left, pRoot.right)

二叉树镜像

  1. # -*- coding:utf-8 -*-
  2. # class TreeNode:
  3. # def __init__(self, x):
  4. # self.val = x
  5. # self.left = None
  6. # self.right = None
  7. class Solution:
  8. # 返回镜像树的根节点
  9. def Mirror(self, root):
  10. if not root:
  11. return None
  12. elif not (root.left or root.right):
  13. return root
  14. root.left, root.right = root.right, root.left
  15. if root.left:
  16. self.Mirror(root.left)
  17. if root.right:
  18. self.Mirror(root.right)

参考

老齐的Github
牛客网

添加新批注
在作者公开此批注前,只有你和作者可见。
回复批注