2018-12-11 15:57:58 +08:00
|
|
|
''' mbinary
|
|
|
|
#########################################################################
|
|
|
|
# File : intervalTree.py
|
|
|
|
# Author: mbinary
|
|
|
|
# Mail: zhuheqin1@gmail.com
|
2019-01-31 12:09:46 +08:00
|
|
|
# Blog: https://mbinary.xyz
|
2018-12-11 15:57:58 +08:00
|
|
|
# Github: https://github.com/mbinary
|
|
|
|
# Created Time: 2018-11-18 10:48
|
|
|
|
# Description:
|
|
|
|
#########################################################################
|
|
|
|
'''
|
2020-04-15 12:28:20 +08:00
|
|
|
from random import randint, shuffle
|
2018-11-07 16:52:55 +08:00
|
|
|
from redBlackTree import redBlackTree
|
|
|
|
|
|
|
|
from functools import total_ordering
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
@total_ordering
|
|
|
|
class node:
|
2020-04-15 12:28:20 +08:00
|
|
|
def __init__(self, low, high, left=None, right=None, isBlack=False):
|
|
|
|
self.val = low # self.val is the low
|
2018-11-07 16:52:55 +08:00
|
|
|
self.high = high
|
|
|
|
self.max = high
|
|
|
|
self.left = left
|
|
|
|
self.right = right
|
2020-04-15 12:28:20 +08:00
|
|
|
self.parent = None
|
2018-11-07 16:52:55 +08:00
|
|
|
self.isBlack = isBlack
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def __lt__(self, nd):
|
2018-11-07 16:52:55 +08:00
|
|
|
return self.val < nd.val
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def __eq__(self, nd):
|
2018-11-07 16:52:55 +08:00
|
|
|
return nd is not None and self.val == nd.val
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def setChild(self, nd, isLeft=True):
|
|
|
|
if isLeft:
|
|
|
|
self.left = nd
|
|
|
|
else:
|
|
|
|
self.right = nd
|
|
|
|
if nd is not None:
|
|
|
|
nd.parent = self
|
|
|
|
|
|
|
|
def getChild(self, isLeft):
|
|
|
|
if isLeft:
|
|
|
|
return self.left
|
|
|
|
else:
|
|
|
|
return self.right
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __bool__(self):
|
|
|
|
return self.val is not None
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __str__(self):
|
|
|
|
color = 'B' if self.isBlack else 'R'
|
|
|
|
return f'{color}[{self.val},{self.high}]-{self.max}'
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def __repr__(self):
|
|
|
|
return f'intervalNode({self.val},{self.high},{self.max},isBlack={self.isBlack})'
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def overlap(self, low, high):
|
|
|
|
return self.val <= high and self.high >= low
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def setMax(self):
|
|
|
|
l = 0 if self.left is None else self.left.max
|
|
|
|
r = 0 if self.right is None else self.right.max
|
|
|
|
self.max = max(self.high, l, r)
|
|
|
|
return self.max
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
class intervalTree(redBlackTree):
|
2020-04-15 12:28:20 +08:00
|
|
|
def search(self, low, high):
|
2018-11-07 16:52:55 +08:00
|
|
|
nd = self.root
|
2020-04-15 12:28:20 +08:00
|
|
|
while nd is not None and not nd.overlap(low, high):
|
|
|
|
if nd.left is not None and nd.left.max >= low:
|
2018-11-07 16:52:55 +08:00
|
|
|
nd = nd.left
|
2020-04-15 12:28:20 +08:00
|
|
|
else:
|
|
|
|
nd = nd.right
|
2018-11-07 16:52:55 +08:00
|
|
|
return nd
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def insert(self, nd):
|
|
|
|
super(intervalTree, self).insert(nd)
|
2018-11-07 16:52:55 +08:00
|
|
|
while nd is not None:
|
|
|
|
nd.setMax()
|
|
|
|
nd = nd.parent
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def delete(self, val):
|
2018-11-07 16:52:55 +08:00
|
|
|
nd = self.find(val)
|
|
|
|
if nd is not None:
|
|
|
|
nd.max = 0
|
|
|
|
tmp = nd.parent
|
|
|
|
while tmp is not None:
|
|
|
|
tmp.setMax()
|
|
|
|
tmp = tmp.parent
|
2020-04-15 12:28:20 +08:00
|
|
|
super(intervalTree, self).delete(val)
|
|
|
|
|
|
|
|
def rotate(self, prt, chd):
|
2018-11-07 16:52:55 +08:00
|
|
|
'''rotate prt, and return new prt, namyly the original chd'''
|
2020-04-15 12:28:20 +08:00
|
|
|
super(intervalTree, self).rotate(prt, chd)
|
2018-11-07 16:52:55 +08:00
|
|
|
prt.setMax()
|
|
|
|
chd.setMax()
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def copyNode(self, src, des):
|
2018-11-07 16:52:55 +08:00
|
|
|
des.val = src.val
|
|
|
|
des.high = src.high
|
|
|
|
des.setMax()
|
|
|
|
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
def genNum(n=10, upper=10):
|
|
|
|
nums = {}
|
2018-11-07 16:52:55 +08:00
|
|
|
for i in range(n):
|
|
|
|
while 1:
|
2020-04-15 12:28:20 +08:00
|
|
|
d = randint(0, 100)
|
2018-11-07 16:52:55 +08:00
|
|
|
if d not in nums:
|
2020-04-15 12:28:20 +08:00
|
|
|
nums[d] = (d, randint(d, d+upper))
|
2018-11-07 16:52:55 +08:00
|
|
|
break
|
|
|
|
return nums.values()
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
def buildTree(n=10, nums=None, visitor=None):
|
2018-12-11 15:28:05 +08:00
|
|
|
#if nums is None or nums ==[]: nums = genNum(n)
|
2018-11-07 16:52:55 +08:00
|
|
|
tree = intervalTree()
|
|
|
|
print(f'build a red-black tree using {nums}')
|
|
|
|
for i in nums:
|
|
|
|
tree.insert(node(*i))
|
|
|
|
if visitor:
|
2020-04-15 12:28:20 +08:00
|
|
|
visitor(tree, i)
|
|
|
|
return tree, nums
|
|
|
|
|
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def testInsert(nums=None):
|
2020-04-15 12:28:20 +08:00
|
|
|
def visitor(t, val):
|
2018-11-07 16:52:55 +08:00
|
|
|
print('inserting', val)
|
|
|
|
print(t)
|
2020-04-15 12:28:20 +08:00
|
|
|
tree, nums = buildTree(visitor=visitor, nums=nums)
|
|
|
|
print('-'*5 + 'in-order visit' + '-'*5)
|
|
|
|
for i, j in enumerate(tree.sort()):
|
2018-11-07 16:52:55 +08:00
|
|
|
print(f'{i+1}: {j}')
|
2018-12-11 15:28:05 +08:00
|
|
|
return tree
|
2018-11-07 16:52:55 +08:00
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def testSuc(nums=None):
|
2020-04-15 12:28:20 +08:00
|
|
|
tree, nums = buildTree(nums=nums)
|
2018-11-07 16:52:55 +08:00
|
|
|
for i in tree.sort():
|
|
|
|
print(f'{i}\'s suc is {tree.getSuccessor(i)}')
|
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
2018-11-07 16:52:55 +08:00
|
|
|
def testDelete(nums=None):
|
2020-04-15 12:28:20 +08:00
|
|
|
tree, nums = buildTree(nums=nums)
|
2018-11-07 16:52:55 +08:00
|
|
|
print(tree)
|
|
|
|
for i in nums:
|
|
|
|
print(f'deleting {i}')
|
|
|
|
tree.delete(i[0])
|
|
|
|
print(tree)
|
2018-12-11 15:28:05 +08:00
|
|
|
return tree
|
2018-11-07 16:52:55 +08:00
|
|
|
|
2020-04-15 12:28:20 +08:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
lst = [(0, 3), (5, 8), (6, 10), (26, 26), (25, 30),
|
|
|
|
(8, 9), (19, 20), (15, 23), (16, 21), (17, 19)]
|
2018-12-11 15:28:05 +08:00
|
|
|
#lst = None
|
2020-04-15 12:28:20 +08:00
|
|
|
# testSuc(lst)
|
2018-12-11 15:28:05 +08:00
|
|
|
tree = testInsert(lst)
|
|
|
|
#tree,_= buildTree(lst)
|
|
|
|
while 1:
|
2020-04-15 12:28:20 +08:00
|
|
|
a = int(input('low:'))
|
|
|
|
b = int(input('high:'))
|
|
|
|
res = tree.search(a, b)
|
2018-12-11 15:28:05 +08:00
|
|
|
print(res)
|