-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathveb.py
131 lines (121 loc) · 4.36 KB
/
veb.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
# The implementation of VEB tree is borrowed from:
# https://github.com/erikwaing/VEBTree
import math
class VEB:
def high(self, x):
return int(math.floor(x / math.sqrt(self.u)))
def low(self, x):
return int((x % math.ceil(math.sqrt(self.u))))
def index(self, x, y):
return int((x * math.floor(math.sqrt(self.u))) + y)
def __init__(self, u):
if u < 0:
raise Exception("u cannot be less than 0 --- u = " + str(u))
self.u = 2
while self.u < u:
self.u *= self.u
self.min = None
self.max = None
if (u > 2):
self.clusters = [None for i in range(self.high(self.u))] # VEB(self.high(self.u))
self.summary = None # VEB(self.high(self.u))
def member(self, x):
if x == self.min or x == self.max: # found it as the minimum or maximum
return True
elif self.u <= 2: # has not found it in the "leaf"
return False
else:
cluster = self.clusters[self.high(x)]
if cluster != None:
return cluster.member(self.low(x)) # looks for it in the clusters inside
else:
return False
def successor(self, x):
if self.u <= 2:
if x == 0 and self.max == 1:
return 1
else:
return None
elif self.min != None and x < self.min: # x is less than everything in the tree, returns the minimum
return self.min
else:
h = self.high(x)
l = self.low(x)
maxlow = None
cluster = self.clusters[h]
if cluster != None:
maxlow = cluster.max
if maxlow != None and l < maxlow:
offset = cluster.successor(l)
return self.index(h, offset)
else:
succcluster = None
if self.summary != None:
succcluster = self.summary.successor(h)
if succcluster == None:
return None
else:
cluster2 = self.clusters[succcluster]
offset = 0
if cluster2 != None:
offset = cluster2.min
return self.index(succcluster, offset)
def predecessor(self, x):
if self.u <= 2:
if x == 1 and self.min == 0:
return 0
else:
return None
elif self.max != None and x > self.max:
return self.max
else:
h = self.high(x)
l = self.low(x)
minlow = None
cluster = self.clusters[h]
if cluster != None:
minlow = cluster.min
if minlow != None and l > minlow:
offset = cluster.predecessor(l)
if offset == None:
offset = 0
return self.index(h, offset)
else:
predcluster = None
if self.summary != None:
predcluster = self.summary.predecessor(h)
if predcluster == None:
if self.min != None and x > self.min:
return self.min
else:
return None
else:
cluster2 = self.clusters[predcluster]
offset = 0
if cluster2 != None:
offset = cluster2.max
return self.index(predcluster, offset)
def emptyInsert(self, x):
self.min = x
self.max = x
def insert(self, x):
if self.min == None:
self.emptyInsert(x)
else:
if x < self.min:
temp = self.min
self.min = x
x = temp
if self.u > 2:
h = self.high(x)
if self.clusters[h] == None:
self.clusters[h] = VEB(self.high(self.u))
if self.summary == None:
self.summary = VEB(self.high(self.u))
if self.clusters[h].min == None:
self.summary.insert(h)
self.clusters[h].emptyInsert(self.low(x))
else:
self.clusters[h].insert(self.low(x))
if x > self.max:
self.max = x