-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathset.lua
executable file
·173 lines (142 loc) · 4.17 KB
/
set.lua
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
#!/usr/bin/env lua
-- Copyright © 2025 Mark Summerfield. All rights reserved.
local ok, lx = pcall(require, "lx.lx")
if not ok then lx = require("lx") end
local Set
local methods = {}
function methods.typeof() return "Set" end
function methods:isempty() return next(self.values_) == nil end
function methods:len() return self.size_ end
function methods:contains(value) return self.values_[value] == true end
function methods:unite(set) -- use :add(a, b, c) to add individual values
for value in pairs(set.values_) do
self:add(value)
end
end
function methods:union(set) -- a | b
local union = Set()
for value in pairs(self.values_) do
union:add(value)
end
for value in pairs(set.values_) do
union:add(value)
end
return union
end
function methods:intersection(set) -- a & b
local intersection = Set()
for value in pairs(self.values_) do
if set:contains(value) then intersection:add(value) end
end
return intersection
end
function methods:difference(set) -- a - b
local difference = Set()
for value in pairs(self.values_) do
if not set:contains(value) then difference:add(value) end
end
return difference
end
function methods:symmetric_difference(set)
local difference = Set()
for value in pairs(self.values_) do
if not set:contains(value) then difference:add(value) end
end
for value in pairs(set.values_) do
if not self:contains(value) then difference:add(value) end
end
return difference
end
function methods:isdisjoint(set)
for value in pairs(self.values_) do
if set:contains(value) then return false end
end
return true
end
function methods:issubset(set)
for value in pairs(self.values_) do
if not set:contains(value) then return false end
end
return true
end
function methods:issuperset(set) return set:issubset(self) end
function methods:random_value()
local index = math.random(1, self.size_)
local i = 1
for value in pairs(self.values_) do
if i == index then return value end
i = i + 1
end
error("Set:random_value() no random value found")
end
function methods:iter(sorted)
local values = self:values(sorted)
local i = 0
return function()
i = i + 1
return values[i]
end
end
function methods:values(sorted)
local values = {}
for value in pairs(self.values_) do
values[#values + 1] = value
end
if sorted then table.sort(values) end
return values
end
function methods:tostring(sorted)
local strs = {}
for value in pairs(self.values_) do
table.insert(strs, lx.dump(value))
end
if sorted then table.sort(strs) end
return "{" .. table.concat(strs, " ") .. "}"
end
function methods:add(...) -- use :unite(set) to add values from another set
for _, value in ipairs({ ... }) do
if not self.values_[value] then
self.values_[value] = true
self.size_ = self.size_ + 1
end
end
end
function methods:remove(...) -- use :difference(set) to remove another set
local removed = 0
for _, value in ipairs({ ... }) do
if self.values_[value] then
self.values_[value] = nil
self.size_ = self.size_ - 1
removed = removed + 1
end
end
return removed
end
function methods:clear()
self.values_ = {}
self.size_ = 0
end
local meta = { __index = methods }
function meta:__tostring() return self:tostring() end
function meta:__len() return self.size_ end
function meta:__eq(set)
if self.size_ ~= #set then return false end
-- using contains means that iteration order doesn't matter
for value in pairs(self.values_) do
if not set:contains(value) then return false end
end
for value in pairs(set.values_) do
if not self:contains(value) then return false end
end
return true
end
function meta:__band(set) return self:intersection(set) end
function meta:__bor(set) return self:union(set) end
function meta:__sub(set) return self:difference(set) end
Set = function(...)
local self = { values_ = {}, size_ = 0 }
setmetatable(self, meta)
self:add(...)
return self
end
return Set