-
Notifications
You must be signed in to change notification settings - Fork 465
/
Copy pathbelt_SetString.ml
201 lines (178 loc) · 5 KB
/
belt_SetString.ml
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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
# 4 "others/belt_Set.cppo.ml"
module I = Belt_internalSetString
# 8 "others/belt_Set.cppo.ml"
module N = Belt_internalAVLset
module A = Belt_Array
type value = I.value
type t = I.t
let empty = None
let isEmpty = N.isEmpty
let minimum = N.minimum
let minUndefined = N.minUndefined
let maximum = N.maximum
let maxUndefined = N.maxUndefined
let forEach = N.forEach
let forEachU = N.forEachU
let reduce = N.reduce
let reduceU = N.reduceU
let every = N.every
let everyU = N.everyU
let some = N.some
let someU = N.someU
let keep = N.keepShared
let keepU = N.keepSharedU
let partition = N.partitionShared
let partitionU = N.partitionSharedU
let size = N.size
let toList = N.toList
let toArray = N.toArray
let fromSortedArrayUnsafe = N.fromSortedArrayUnsafe
let checkInvariantInternal = N.checkInvariantInternal
let rec add (t : t) (x : value) : t =
match t with
None -> N.singleton x
| Some nt ->
let v = nt.value in
if x = v then t else
let {N.left = l; right = r} = nt in
if x < v then
let ll = add l x in
if ll == l then t
else N.bal ll v r
else
let rr = add r x in
if rr == r then t
else N.bal l v rr
let mergeMany h arr =
let len = A.length arr in
let v = ref h in
for i = 0 to len - 1 do
let key = A.getUnsafe arr i in
v .contents<- add v.contents key
done ;
v.contents
let rec remove (t : t) (x : value) : t =
match t with
| None -> t
| Some n ->
let {N.left = l; value = v; right = r} = n in
if x = v then
match l, r with
| None, _ -> r
| _, None -> l
| _, Some rn ->
let v = ref rn.value in
let r = N.removeMinAuxWithRef rn v in
N.bal l v.contents r
else
if x < v then
let ll = remove l x in
if ll == l then t
else N.bal ll v r
else
let rr = remove r x in
if rr == r then t
else N.bal l v rr
let removeMany h arr =
let len = A.length arr in
let v = ref h in
for i = 0 to len - 1 do
let key = A.getUnsafe arr i in
v .contents<- remove v.contents key
done ;
v.contents
let fromArray = I.fromArray
let cmp = I.cmp
let eq = I.eq
let get = I.get
let getUndefined = I.getUndefined
let getExn = I.getExn
let subset = I.subset
let has = I.has
let rec splitAuxNoPivot (n : _ N.node) (x : value) : t * t =
let {N.left = l; value = v; right = r} = n in
if x = v then l, r
else if x < v then
match l with
| None ->
None , Some n
| Some l ->
let ll, rl = splitAuxNoPivot l x in
ll, N.joinShared rl v r
else
match r with
| None ->
Some n, None
| Some r ->
let lr, rr = splitAuxNoPivot r x in
N.joinShared l v lr, rr
let rec splitAuxPivot (n : _ N.node) (x : value) pres : t * t =
let {N.left = l; value = v; right = r} = n in
if x = v then begin
pres .contents<- true;
(l, r)
end
else if x < v then
match l with
| None ->
None, Some n
| Some l ->
let ll, rl = splitAuxPivot l x pres in
ll, N.joinShared rl v r
else
match r with
| None ->
Some n, None
| Some r ->
let lr, rr = splitAuxPivot r x pres in
N.joinShared l v lr, rr
let split (t : t) (x : value) =
match t with
None ->
(None, None), false
| Some n ->
let pres = ref false in
let v = splitAuxPivot n x pres in
v, pres.contents
let rec union (s1 : t) (s2 : t) =
match s1, s2 with
(None, _) -> s2
| (_, None) -> s1
| Some n1, Some n2 (* (Node(l1, v1, r1, h1), Node(l2, v2, r2, h2)) *) ->
let h1, h2 = n1.height, n2.height in
if h1 >= h2 then
if h2 = 1 then add s1 n2.value else begin
let {N.left = l1; value = v1; right = r1} = n1 in
let (l2, r2) = splitAuxNoPivot n2 v1 in
N.joinShared (union l1 l2) v1 (union r1 r2)
end
else
if h1 = 1 then add s2 n1.value else begin
let {N.left = l2; value = v2; right = r2} = n2 in
let (l1, r1) = splitAuxNoPivot n1 v2 in
N.joinShared (union l1 l2) v2 (union r1 r2)
end
let rec intersect (s1 : t) (s2 : t) =
match s1, s2 with
(None, _)
| (_, None) -> None
| Some n1, Some n2 (* (Node(l1, v1, r1, _), t2) *) ->
let {N.left = l1; value = v1; right = r1} = n1 in
let pres = ref false in
let l2,r2 = splitAuxPivot n2 v1 pres in
let ll = intersect l1 l2 in
let rr = intersect r1 r2 in
if pres.contents then N.joinShared ll v1 rr
else N.concatShared ll rr
let rec diff (s1 : t) (s2 : t) =
match s1, s2 with
| (None, _)
| (_, None) -> s1
| Some n1, Some n2 (* (Node(l1, v1, r1, _), t2) *) ->
let {N.left = l1; value = v1; right = r1} = n1 in
let pres = ref false in
let l2, r2 = splitAuxPivot n2 v1 pres in
let ll = diff l1 l2 in
let rr = diff r1 r2 in
if pres.contents then N.concatShared ll rr
else N.joinShared ll v1 rr