Added inverse dicts and a few new list methods.
This commit is contained in:
parent
11997f0255
commit
84f3ae76ab
172
containers.lua
172
containers.lua
@ -1,4 +1,4 @@
|
|||||||
local List, Dict
|
local List, Dict, Undict, _undict_mt, _dict_mt
|
||||||
local insert, remove, concat
|
local insert, remove, concat
|
||||||
do
|
do
|
||||||
local _obj_0 = table
|
local _obj_0 = table
|
||||||
@ -226,6 +226,23 @@ local _list_mt = {
|
|||||||
return _accum_0
|
return _accum_0
|
||||||
end)())
|
end)())
|
||||||
end,
|
end,
|
||||||
|
copy = function(self)
|
||||||
|
return List((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for i = 1, #self do
|
||||||
|
_accum_0[_len_0] = self[i]
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)())
|
||||||
|
end,
|
||||||
|
reverse = function(self)
|
||||||
|
local n = #self
|
||||||
|
for i = 1, math.floor(n / 2) do
|
||||||
|
self[i], self[n - i + 1] = self[n - i + 1], self[i]
|
||||||
|
end
|
||||||
|
end,
|
||||||
reversed = function(self)
|
reversed = function(self)
|
||||||
return List((function()
|
return List((function()
|
||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
@ -236,6 +253,49 @@ local _list_mt = {
|
|||||||
end
|
end
|
||||||
return _accum_0
|
return _accum_0
|
||||||
end)())
|
end)())
|
||||||
|
end,
|
||||||
|
sort = function(self)
|
||||||
|
return table.sort(self)
|
||||||
|
end,
|
||||||
|
sort_by = function(self, fn)
|
||||||
|
local keys = setmetatable({ }, {
|
||||||
|
__index = function(self, k)
|
||||||
|
local key = fn(k)
|
||||||
|
self[k] = key
|
||||||
|
return key
|
||||||
|
end
|
||||||
|
})
|
||||||
|
return table.sort(self, function(a, b)
|
||||||
|
return keys[a] <= keys[b]
|
||||||
|
end)
|
||||||
|
end,
|
||||||
|
sorted = function(self)
|
||||||
|
local c = self:copy()
|
||||||
|
c:sort()
|
||||||
|
return c
|
||||||
|
end,
|
||||||
|
sorted_by = function(self, fn)
|
||||||
|
local c = self:copy()
|
||||||
|
c:sort_by(fn)
|
||||||
|
return c
|
||||||
|
end,
|
||||||
|
filter_by = function(self, keep)
|
||||||
|
local deleted = 0
|
||||||
|
for i = 1, #self do
|
||||||
|
if not (keep(self[i])) then
|
||||||
|
deleted = deleted + 1
|
||||||
|
elseif deleted > 0 then
|
||||||
|
self[i - deleted] = self[i]
|
||||||
|
end
|
||||||
|
end
|
||||||
|
for i = #self - deleted + 1, #self do
|
||||||
|
self[i] = nil
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
filtered_by = function(self, keep)
|
||||||
|
local c = self:copy()
|
||||||
|
c:filter_by(keep)
|
||||||
|
return c
|
||||||
end
|
end
|
||||||
},
|
},
|
||||||
__newindex = function(self, k, v)
|
__newindex = function(self, k, v)
|
||||||
@ -262,7 +322,91 @@ List = function(t)
|
|||||||
return error("Unsupported List type: " .. type(t))
|
return error("Unsupported List type: " .. type(t))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
local _dict_mt = {
|
local compliments = setmetatable({ }, {
|
||||||
|
__mode = 'k'
|
||||||
|
})
|
||||||
|
_undict_mt = {
|
||||||
|
__index = function(self, k)
|
||||||
|
return not compliments[self][k] and true or nil
|
||||||
|
end,
|
||||||
|
__newindex = function(self, k, v)
|
||||||
|
if k then
|
||||||
|
compliments[self][k] = nil
|
||||||
|
else
|
||||||
|
compliments[self][k] = true
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
__eq = function(self, other)
|
||||||
|
if not (type(other) == 'table' and getmetatable(other) == getmetatable(self)) then
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
return compliments[self] == compliments[other]
|
||||||
|
end,
|
||||||
|
__len = function(self)
|
||||||
|
return math.huge
|
||||||
|
end,
|
||||||
|
__tostring = function(self)
|
||||||
|
return "~" .. _dict_mt.__tostring(compliments[self])
|
||||||
|
end,
|
||||||
|
as_nomsu = function(self)
|
||||||
|
return "~" .. _dict_mt.as_nomsu(compliments[self])
|
||||||
|
end,
|
||||||
|
as_lua = function(self)
|
||||||
|
return "~" .. __dict_mt.as_lua(compliments[self])
|
||||||
|
end,
|
||||||
|
__band = function(self, other)
|
||||||
|
if getmetatable(other) == _undict_mt then
|
||||||
|
return Undict(_dict_mt.__bor(compliments[self], compliments[other]))
|
||||||
|
else
|
||||||
|
return _dict_mt.__band(other, self)
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
__bor = function(self, other)
|
||||||
|
if getmetatable(other) == _undict_mt then
|
||||||
|
return Undict(_dict_mt.__band(compliments[self], compliments[other]))
|
||||||
|
else
|
||||||
|
return Undict((function()
|
||||||
|
local _tbl_0 = { }
|
||||||
|
for k, v in pairs(compliments[self]) do
|
||||||
|
if not other[k] then
|
||||||
|
_tbl_0[k] = v
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return _tbl_0
|
||||||
|
end)())
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
__bxor = function(self, other)
|
||||||
|
if getmetatable(other) == _undict_mt then
|
||||||
|
return _dict_mt.__bxor(compliments[self], compliments[other])
|
||||||
|
else
|
||||||
|
return Undict(_dict_mt.__band(other, self))
|
||||||
|
end
|
||||||
|
end,
|
||||||
|
__bnot = function(self)
|
||||||
|
return Dict((function()
|
||||||
|
local _tbl_0 = { }
|
||||||
|
for k, v in pairs(compliments[self]) do
|
||||||
|
_tbl_0[k] = v
|
||||||
|
end
|
||||||
|
return _tbl_0
|
||||||
|
end)())
|
||||||
|
end
|
||||||
|
}
|
||||||
|
Undict = function(d)
|
||||||
|
local u = setmetatable({ }, _undict_mt)
|
||||||
|
compliments[u] = Dict((function()
|
||||||
|
local _tbl_0 = { }
|
||||||
|
for k, v in pairs(d) do
|
||||||
|
if v then
|
||||||
|
_tbl_0[k] = true
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return _tbl_0
|
||||||
|
end)())
|
||||||
|
return u
|
||||||
|
end
|
||||||
|
_dict_mt = {
|
||||||
__type = "a Dict",
|
__type = "a Dict",
|
||||||
__eq = function(self, other)
|
__eq = function(self, other)
|
||||||
if not (type(other) == 'table' and getmetatable(other) == getmetatable(self)) then
|
if not (type(other) == 'table' and getmetatable(other) == getmetatable(self)) then
|
||||||
@ -292,7 +436,7 @@ local _dict_mt = {
|
|||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
_accum_0[_len_0] = "." .. tostring(k) .. " = " .. tostring(v)
|
_accum_0[_len_0] = v == true and "." .. as_nomsu(k) or "." .. tostring(k) .. " = " .. tostring(v)
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
return _accum_0
|
return _accum_0
|
||||||
@ -303,7 +447,7 @@ local _dict_mt = {
|
|||||||
local _accum_0 = { }
|
local _accum_0 = { }
|
||||||
local _len_0 = 1
|
local _len_0 = 1
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
_accum_0[_len_0] = "." .. tostring(as_nomsu(k)) .. " = " .. tostring(as_nomsu(v))
|
_accum_0[_len_0] = v == true and "." .. as_nomsu(k) or "." .. tostring(as_nomsu(k)) .. " = " .. tostring(as_nomsu(v))
|
||||||
_len_0 = _len_0 + 1
|
_len_0 = _len_0 + 1
|
||||||
end
|
end
|
||||||
return _accum_0
|
return _accum_0
|
||||||
@ -320,11 +464,22 @@ local _dict_mt = {
|
|||||||
return _accum_0
|
return _accum_0
|
||||||
end)(), ", ") .. "}"
|
end)(), ", ") .. "}"
|
||||||
end,
|
end,
|
||||||
|
as_list = function(self)
|
||||||
|
return List((function()
|
||||||
|
local _accum_0 = { }
|
||||||
|
local _len_0 = 1
|
||||||
|
for k, v in pairs(self) do
|
||||||
|
_accum_0[_len_0] = k
|
||||||
|
_len_0 = _len_0 + 1
|
||||||
|
end
|
||||||
|
return _accum_0
|
||||||
|
end)())
|
||||||
|
end,
|
||||||
__band = function(self, other)
|
__band = function(self, other)
|
||||||
return Dict((function()
|
return Dict((function()
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
if other[k] ~= nil then
|
if other[k] then
|
||||||
_tbl_0[k] = v
|
_tbl_0[k] = v
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@ -332,6 +487,9 @@ local _dict_mt = {
|
|||||||
end)())
|
end)())
|
||||||
end,
|
end,
|
||||||
__bor = function(self, other)
|
__bor = function(self, other)
|
||||||
|
if getmetatable(other) == _undict_mt then
|
||||||
|
return _undict_mt.__bor(other, self)
|
||||||
|
end
|
||||||
local ret = Dict((function()
|
local ret = Dict((function()
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
@ -347,6 +505,9 @@ local _dict_mt = {
|
|||||||
return ret
|
return ret
|
||||||
end,
|
end,
|
||||||
__bxor = function(self, other)
|
__bxor = function(self, other)
|
||||||
|
if getmetatable(other) == _undict_mt then
|
||||||
|
return _undict_mt.__bxor(other, self)
|
||||||
|
end
|
||||||
local ret = Dict((function()
|
local ret = Dict((function()
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
for k, v in pairs(self) do
|
for k, v in pairs(self) do
|
||||||
@ -363,6 +524,7 @@ local _dict_mt = {
|
|||||||
end
|
end
|
||||||
return ret
|
return ret
|
||||||
end,
|
end,
|
||||||
|
__bnot = Undict,
|
||||||
__add = function(self, other)
|
__add = function(self, other)
|
||||||
local ret = Dict((function()
|
local ret = Dict((function()
|
||||||
local _tbl_0 = { }
|
local _tbl_0 = { }
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
-- This file contains container classes, i.e. Lists and Dicts, plus some extended string functionality
|
-- This file contains container classes, i.e. Lists and Dicts, plus some extended string functionality
|
||||||
local List, Dict
|
local List, Dict, Undict, _undict_mt, _dict_mt
|
||||||
{:insert,:remove,:concat} = table
|
{:insert,:remove,:concat} = table
|
||||||
|
|
||||||
as_nomsu = =>
|
as_nomsu = =>
|
||||||
@ -88,8 +88,42 @@ _list_mt =
|
|||||||
start = (n+1-start) if start < 0
|
start = (n+1-start) if start < 0
|
||||||
stop = (n+1-stop) if stop < 0
|
stop = (n+1-stop) if stop < 0
|
||||||
return List[@[i] for i=start,stop]
|
return List[@[i] for i=start,stop]
|
||||||
|
copy: => List[@[i] for i=1,#@]
|
||||||
|
reverse: =>
|
||||||
|
n = #@
|
||||||
|
for i=1,math.floor(n/2)
|
||||||
|
@[i], @[n-i+1] = @[n-i+1], @[i]
|
||||||
reversed: => List[@[i] for i=#@,1,-1]
|
reversed: => List[@[i] for i=#@,1,-1]
|
||||||
|
sort: => table.sort(@)
|
||||||
|
sort_by: (fn)=>
|
||||||
|
keys = setmetatable {},
|
||||||
|
__index: (k)=>
|
||||||
|
key = fn(k)
|
||||||
|
@[k] = key
|
||||||
|
return key
|
||||||
|
table.sort(@, (a,b)->keys[a] <= keys[b])
|
||||||
|
sorted: =>
|
||||||
|
c = @copy!
|
||||||
|
c\sort!
|
||||||
|
return c
|
||||||
|
sorted_by: (fn)=>
|
||||||
|
c = @copy!
|
||||||
|
c\sort_by(fn)
|
||||||
|
return c
|
||||||
|
filter_by: (keep)=>
|
||||||
|
deleted = 0
|
||||||
|
for i=1,#@
|
||||||
|
unless keep(@[i])
|
||||||
|
deleted += 1
|
||||||
|
elseif deleted > 0
|
||||||
|
@[i-deleted] = @[i]
|
||||||
|
for i=#@-deleted+1,#@
|
||||||
|
@[i] = nil
|
||||||
|
|
||||||
|
filtered_by: (keep)=>
|
||||||
|
c = @copy!
|
||||||
|
c\filter_by(keep)
|
||||||
|
return c
|
||||||
|
|
||||||
-- TODO: remove this safety check to get better performance?
|
-- TODO: remove this safety check to get better performance?
|
||||||
__newindex: (k,v)=>
|
__newindex: (k,v)=>
|
||||||
@ -109,6 +143,52 @@ List = (t)->
|
|||||||
return l
|
return l
|
||||||
else error("Unsupported List type: "..type(t))
|
else error("Unsupported List type: "..type(t))
|
||||||
|
|
||||||
|
|
||||||
|
compliments = setmetatable({}, {__mode:'k'})
|
||||||
|
_undict_mt =
|
||||||
|
__type: "an Inverse Dict"
|
||||||
|
__index: (k)=> not compliments[@][k] and true or nil
|
||||||
|
__newindex: (k,v)=>
|
||||||
|
if k
|
||||||
|
compliments[@][k] = nil
|
||||||
|
else
|
||||||
|
compliments[@][k] = true
|
||||||
|
__eq: (other)=>
|
||||||
|
unless type(other) == 'table' and getmetatable(other) == getmetatable(@)
|
||||||
|
return false
|
||||||
|
return compliments[@] == compliments[other]
|
||||||
|
__len: => math.huge
|
||||||
|
__tostring: => "~".._dict_mt.__tostring(compliments[@])
|
||||||
|
as_nomsu: => "~".._dict_mt.as_nomsu(compliments[@])
|
||||||
|
as_lua: => "~"..__dict_mt.as_lua(compliments[@])
|
||||||
|
__band: (other)=>
|
||||||
|
if getmetatable(other) == _undict_mt
|
||||||
|
-- ~{x,y} & ~{y,z} == ~{x,y,z} == ~({x,y} | {y,z})
|
||||||
|
Undict(_dict_mt.__bor(compliments[@], compliments[other]))
|
||||||
|
else
|
||||||
|
-- ~{x,y} & {y,z} == {z} == {y,z} & ~{x,y}
|
||||||
|
_dict_mt.__band(other, @)
|
||||||
|
__bor: (other)=>
|
||||||
|
if getmetatable(other) == _undict_mt
|
||||||
|
-- ~{x,y} | ~{y,z} == ~{y} = ~({x,y} & {y,z})
|
||||||
|
Undict(_dict_mt.__band(compliments[@], compliments[other]))
|
||||||
|
else
|
||||||
|
-- ~{x,y} | {y,z} == ~{z} = ~({y,z} & ~{x,y})
|
||||||
|
Undict{k,v for k,v in pairs(compliments[@]) when not other[k]}
|
||||||
|
__bxor: (other)=>
|
||||||
|
if getmetatable(other) == _undict_mt
|
||||||
|
-- ~{x,y} ^ ~{y,z} == {x,z} = {x,y} ^ {y,z}
|
||||||
|
_dict_mt.__bxor(compliments[@], compliments[other])
|
||||||
|
else
|
||||||
|
-- ~{x,y} ^ {y,z} == ~{x} = ~({x,y} & ~{y,z})
|
||||||
|
Undict(_dict_mt.__band(other, @))
|
||||||
|
__bnot: => Dict{k,v for k,v in pairs(compliments[@])}
|
||||||
|
|
||||||
|
Undict = (d)->
|
||||||
|
u = setmetatable({}, _undict_mt)
|
||||||
|
compliments[u] = Dict{k,true for k,v in pairs(d) when v}
|
||||||
|
return u
|
||||||
|
|
||||||
_dict_mt =
|
_dict_mt =
|
||||||
__type: "a Dict"
|
__type: "a Dict"
|
||||||
__eq: (other)=>
|
__eq: (other)=>
|
||||||
@ -124,24 +204,31 @@ _dict_mt =
|
|||||||
for _ in pairs(@) do n += 1
|
for _ in pairs(@) do n += 1
|
||||||
return n
|
return n
|
||||||
__tostring: =>
|
__tostring: =>
|
||||||
"{"..concat([".#{k} = #{v}" for k,v in pairs @], ", ").."}"
|
"{"..concat([v == true and "."..as_nomsu(k) or ".#{k} = #{v}" for k,v in pairs @], ", ").."}"
|
||||||
as_nomsu: =>
|
as_nomsu: =>
|
||||||
"{"..concat([".#{as_nomsu(k)} = #{as_nomsu(v)}" for k,v in pairs @], ", ").."}"
|
"{"..concat([v == true and "."..as_nomsu(k) or ".#{as_nomsu(k)} = #{as_nomsu(v)}" for k,v in pairs @], ", ").."}"
|
||||||
as_lua: =>
|
as_lua: =>
|
||||||
"a_Dict{"..concat(["[ #{as_lua(k)}]= #{as_lua(v)}" for k,v in pairs @], ", ").."}"
|
"a_Dict{"..concat(["[ #{as_lua(k)}]= #{as_lua(v)}" for k,v in pairs @], ", ").."}"
|
||||||
|
as_list: => List[k for k,v in pairs(@)]
|
||||||
__band: (other)=>
|
__band: (other)=>
|
||||||
Dict{k,v for k,v in pairs(@) when other[k] != nil}
|
Dict{k,v for k,v in pairs(@) when other[k]}
|
||||||
__bor: (other)=>
|
__bor: (other)=>
|
||||||
|
if getmetatable(other) == _undict_mt
|
||||||
|
return _undict_mt.__bor(other, @)
|
||||||
ret = Dict{k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
return ret
|
return ret
|
||||||
__bxor: (other)=>
|
__bxor: (other)=>
|
||||||
|
if getmetatable(other) == _undict_mt
|
||||||
|
return _undict_mt.__bxor(other, @)
|
||||||
ret = Dict{k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
if ret[k] == nil then ret[k] = v
|
if ret[k] == nil then ret[k] = v
|
||||||
else ret[k] = nil
|
else ret[k] = nil
|
||||||
return ret
|
return ret
|
||||||
|
__bnot: Undict
|
||||||
|
|
||||||
__add: (other)=>
|
__add: (other)=>
|
||||||
ret = Dict{k,v for k,v in pairs(@)}
|
ret = Dict{k,v for k,v in pairs(@)}
|
||||||
for k,v in pairs(other)
|
for k,v in pairs(other)
|
||||||
@ -154,6 +241,7 @@ _dict_mt =
|
|||||||
if ret[k] == nil then ret[k] = -v
|
if ret[k] == nil then ret[k] = -v
|
||||||
else ret[k] -= v
|
else ret[k] -= v
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
Dict = (t)->
|
Dict = (t)->
|
||||||
if type(t) == 'table'
|
if type(t) == 'table'
|
||||||
return setmetatable(t, _dict_mt)
|
return setmetatable(t, _dict_mt)
|
||||||
|
Loading…
Reference in New Issue
Block a user