initial commit: import bigint
This commit is contained in:
commit
cf9d0892bf
48
.gitignore
vendored
Normal file
48
.gitignore
vendored
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
|
||||||
|
# Created by https://www.gitignore.io/api/lua
|
||||||
|
# Edit at https://www.gitignore.io/?templates=lua
|
||||||
|
|
||||||
|
### Lua ###
|
||||||
|
# Compiled Lua sources
|
||||||
|
luac.out
|
||||||
|
|
||||||
|
# luarocks build files
|
||||||
|
*.src.rock
|
||||||
|
*.zip
|
||||||
|
*.tar.gz
|
||||||
|
|
||||||
|
# Object files
|
||||||
|
*.o
|
||||||
|
*.os
|
||||||
|
*.ko
|
||||||
|
*.obj
|
||||||
|
*.elf
|
||||||
|
|
||||||
|
# Precompiled Headers
|
||||||
|
*.gch
|
||||||
|
*.pch
|
||||||
|
|
||||||
|
# Libraries
|
||||||
|
*.lib
|
||||||
|
*.a
|
||||||
|
*.la
|
||||||
|
*.lo
|
||||||
|
*.def
|
||||||
|
*.exp
|
||||||
|
|
||||||
|
# Shared objects (inc. Windows DLLs)
|
||||||
|
*.dll
|
||||||
|
*.so
|
||||||
|
*.so.*
|
||||||
|
*.dylib
|
||||||
|
|
||||||
|
# Executables
|
||||||
|
*.exe
|
||||||
|
*.out
|
||||||
|
*.app
|
||||||
|
*.i*86
|
||||||
|
*.x86_64
|
||||||
|
*.hex
|
||||||
|
|
||||||
|
|
||||||
|
# End of https://www.gitignore.io/api/lua
|
384
bigint.lua
Normal file
384
bigint.lua
Normal file
@ -0,0 +1,384 @@
|
|||||||
|
BigInt = {}
|
||||||
|
BigInt.__index = BigInt
|
||||||
|
|
||||||
|
|
||||||
|
local function trim(a)
|
||||||
|
for i = #a, 2, -1 do
|
||||||
|
if a[i] ~= 0 then break end
|
||||||
|
table.remove(a)
|
||||||
|
end
|
||||||
|
if #a == 1 and a[1] == 0 then a.sign = 1 end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function num2tab(n)
|
||||||
|
local t = {sign=1}
|
||||||
|
local i = 1
|
||||||
|
if n < 0 then
|
||||||
|
n = -n
|
||||||
|
t.sign = -1
|
||||||
|
end
|
||||||
|
repeat
|
||||||
|
local r = math.fmod(n, 10)
|
||||||
|
n = math.floor(n / 10)
|
||||||
|
t[i] = r
|
||||||
|
i = i + 1
|
||||||
|
until n == 0
|
||||||
|
return t
|
||||||
|
end
|
||||||
|
|
||||||
|
local function str2tab(s)
|
||||||
|
local s = string.match(s, "^[+-]?%d+$") or "0"
|
||||||
|
local t = {sign=1}
|
||||||
|
local i = #s
|
||||||
|
local j = 1
|
||||||
|
local first = 1
|
||||||
|
if s:sub(1, 1) == '-' then t.sign=-1;first=2 end
|
||||||
|
if s:sub(1, 1) == '+' then first=2 end
|
||||||
|
while i >= first do
|
||||||
|
t[j] = s:sub(i, i):byte()-48
|
||||||
|
j = j + 1
|
||||||
|
i = i - 1
|
||||||
|
end
|
||||||
|
return t
|
||||||
|
end
|
||||||
|
|
||||||
|
local function iterZip(a, b)
|
||||||
|
local a = BigInt(a)
|
||||||
|
local b = BigInt(b)
|
||||||
|
local i = 1
|
||||||
|
return function()
|
||||||
|
if not a[i] and not b[i] then return end
|
||||||
|
local ai = a[i] or 0
|
||||||
|
local bi = b[i] or 0
|
||||||
|
i = i + 1
|
||||||
|
return ai, bi, i-1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function iterZipRev(a, b, n)
|
||||||
|
local a = BigInt(a)
|
||||||
|
local b = BigInt(b)
|
||||||
|
local i = n
|
||||||
|
if not i then
|
||||||
|
i = #a
|
||||||
|
if #b > i then i = #b end
|
||||||
|
end
|
||||||
|
return function()
|
||||||
|
if i<1 then return end
|
||||||
|
local ai = a[i] or 0
|
||||||
|
local bi = b[i] or 0
|
||||||
|
i = i - 1
|
||||||
|
return ai, bi, i+1
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
local function concat(a, b)
|
||||||
|
if not b then return a:clone() end
|
||||||
|
local c = b:clone()
|
||||||
|
local n = #c
|
||||||
|
for i in ipairs(a) do
|
||||||
|
n = n + 1
|
||||||
|
c[n] = a[i]
|
||||||
|
end
|
||||||
|
c.sign = a.sign
|
||||||
|
trim(c)
|
||||||
|
return c
|
||||||
|
end
|
||||||
|
|
||||||
|
local function append(a, d)
|
||||||
|
table.insert(a, 1, d)
|
||||||
|
trim(a)
|
||||||
|
end
|
||||||
|
|
||||||
|
local function uadd(a, b)
|
||||||
|
local c = BigInt()
|
||||||
|
local carry = 0
|
||||||
|
for ai, bi, i in iterZip(a, b) do
|
||||||
|
local ci = ai + bi + carry
|
||||||
|
if ci > 9 then
|
||||||
|
ci = ci - 10
|
||||||
|
carry = 1
|
||||||
|
else
|
||||||
|
carry = 0
|
||||||
|
end
|
||||||
|
c[i] = ci
|
||||||
|
end
|
||||||
|
trim(c)
|
||||||
|
return c
|
||||||
|
end
|
||||||
|
|
||||||
|
local function ult(a, b)
|
||||||
|
for ai, bi in iterZipRev(a, b) do
|
||||||
|
if ai < bi then return true end
|
||||||
|
if bi < ai then return false end
|
||||||
|
end
|
||||||
|
return false
|
||||||
|
end
|
||||||
|
|
||||||
|
local function usub(a, b)
|
||||||
|
local c = BigInt()
|
||||||
|
if ult(a, b) then a, b = b, a; c.sign = -1 end
|
||||||
|
local borrow = 0
|
||||||
|
local ci
|
||||||
|
for ai, bi, i in iterZip(a, b) do
|
||||||
|
ci = (ai - borrow) - bi
|
||||||
|
if ci < 0 then
|
||||||
|
-- ci = 10 + ai - borrow - bi
|
||||||
|
ci = ci + 10
|
||||||
|
borrow = 1
|
||||||
|
else
|
||||||
|
borrow = 0
|
||||||
|
end
|
||||||
|
c[i] = ci
|
||||||
|
end
|
||||||
|
trim(c)
|
||||||
|
return c
|
||||||
|
end
|
||||||
|
|
||||||
|
local function udiv(a, b)
|
||||||
|
local a = BigInt(a):clone()
|
||||||
|
local b = BigInt(b):clone()
|
||||||
|
local q = BigInt()
|
||||||
|
local d = BigInt()
|
||||||
|
local r
|
||||||
|
a.sign = 1
|
||||||
|
b.sign = 1
|
||||||
|
if b == q then return end
|
||||||
|
if a == q then return q, q end
|
||||||
|
-- if a < b then return q, a:clone() end
|
||||||
|
if a < b then return q, a end
|
||||||
|
if a == b then return BigInt(1), q end
|
||||||
|
-- if #b == 1 and b[1] == 1 then return a:clone(), q end
|
||||||
|
if #b == 1 and b[1] == 1 then return a, q end
|
||||||
|
for k = #a, 1, -1 do
|
||||||
|
-- d:append(a[k])
|
||||||
|
append(d, a[k])
|
||||||
|
if d >= b then
|
||||||
|
for i = 10, 1, -1 do
|
||||||
|
r = d - i * b
|
||||||
|
if r.sign == 1 then
|
||||||
|
-- q = q:concat(BigInt(i))
|
||||||
|
q = concat(q, BigInt(i))
|
||||||
|
break
|
||||||
|
end
|
||||||
|
end
|
||||||
|
d = r
|
||||||
|
else
|
||||||
|
q = q * 10
|
||||||
|
end
|
||||||
|
end
|
||||||
|
return q, r
|
||||||
|
end
|
||||||
|
|
||||||
|
local function uinc(a)
|
||||||
|
-- in place
|
||||||
|
local i = 1
|
||||||
|
local c = 1
|
||||||
|
repeat
|
||||||
|
if not a[i] then a[i] = 0 end
|
||||||
|
a[i] = a[i] + c
|
||||||
|
c = 0
|
||||||
|
if a[i] > 9 then
|
||||||
|
a[i] = a[i] - 10
|
||||||
|
c = 1
|
||||||
|
end
|
||||||
|
i = i + 1
|
||||||
|
until c == 0
|
||||||
|
end
|
||||||
|
|
||||||
|
local function udec(a)
|
||||||
|
-- in place
|
||||||
|
if #a == 1 and a[1] == 0 then
|
||||||
|
a[1] = 1
|
||||||
|
a.sign = -1
|
||||||
|
return
|
||||||
|
end
|
||||||
|
local i = 1
|
||||||
|
local c = 1
|
||||||
|
repeat
|
||||||
|
a[i] = a[i] - c
|
||||||
|
c = 0
|
||||||
|
if a[i] < 0 then
|
||||||
|
a[i] = a[i] + 10
|
||||||
|
c = 1
|
||||||
|
end
|
||||||
|
i = i + 1
|
||||||
|
until c == 0
|
||||||
|
trim(a)
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__eq(a, b)
|
||||||
|
if #a ~= #b then return false end
|
||||||
|
if a.sign ~= b.sign then return false end
|
||||||
|
local i = 1
|
||||||
|
local r = true
|
||||||
|
while a[i] do
|
||||||
|
if a[i] ~= b[i] then
|
||||||
|
r = false
|
||||||
|
break
|
||||||
|
end
|
||||||
|
i = i + 1
|
||||||
|
end
|
||||||
|
return r
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__lt(a, b)
|
||||||
|
return BigInt.__sub(a, b).sign == -1
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__unm(a)
|
||||||
|
local b = BigInt.clone(a)
|
||||||
|
b.sign = a.sign * -1
|
||||||
|
return b
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__add(a, b)
|
||||||
|
local a = BigInt(a)
|
||||||
|
local b = BigInt(b)
|
||||||
|
if a.sign == 1 and b.sign == 1 then
|
||||||
|
return uadd(a, b)
|
||||||
|
elseif a.sign == 1 and b.sign == -1 then
|
||||||
|
return usub(a, b)
|
||||||
|
elseif a.sign == -1 and b.sign == 1 then
|
||||||
|
return usub(b, a)
|
||||||
|
else
|
||||||
|
local r = uadd(a, b)
|
||||||
|
r.sign = -1
|
||||||
|
return r
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__sub(a, b)
|
||||||
|
local a = BigInt(a)
|
||||||
|
local b = BigInt(b)
|
||||||
|
if a.sign == 1 and b.sign == 1 then
|
||||||
|
return usub(a, b)
|
||||||
|
elseif a.sign == 1 and b.sign == -1 then
|
||||||
|
return uadd(a, b)
|
||||||
|
elseif a.sign == -1 and b.sign == 1 then
|
||||||
|
local c = uadd(a, b)
|
||||||
|
c.sign = -1
|
||||||
|
return c
|
||||||
|
else
|
||||||
|
return usub(b, a)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__mul(a, b)
|
||||||
|
local a = BigInt(a)
|
||||||
|
local b = BigInt(b)
|
||||||
|
local c = BigInt()
|
||||||
|
for i = 1, #a do
|
||||||
|
local carry = 0
|
||||||
|
for j = 1, #b do
|
||||||
|
local k = i + j - 1
|
||||||
|
local ai = a[i] or 0
|
||||||
|
local bj = b[j] or 0
|
||||||
|
local ck = c[k] or 0
|
||||||
|
ck = ck + carry + ai * bj
|
||||||
|
-- carry = ck // 10
|
||||||
|
-- c[k] = ck % 10
|
||||||
|
carry = math.floor(ck / 10)
|
||||||
|
c[k] = math.fmod(ck, 10)
|
||||||
|
end
|
||||||
|
c[i+#b] = carry
|
||||||
|
end
|
||||||
|
c.sign = a.sign * b.sign
|
||||||
|
trim(c)
|
||||||
|
return c
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__div(a, b)
|
||||||
|
local q = udiv(a, b)
|
||||||
|
q.sign = a.sign * b.sign
|
||||||
|
trim(q)
|
||||||
|
return q
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.__pow(a, b)
|
||||||
|
local a = BigInt(a)
|
||||||
|
local b = BigInt(b):clone()
|
||||||
|
local c = BigInt(1)
|
||||||
|
local i = BigInt(1)
|
||||||
|
if b.sign == -1 then b.sign = 1 end
|
||||||
|
while i <= b do
|
||||||
|
c = c * a
|
||||||
|
i:inc()
|
||||||
|
end
|
||||||
|
return c
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt:__tostring()
|
||||||
|
local l = {}
|
||||||
|
local i = 1
|
||||||
|
-- trim(self)
|
||||||
|
-- if #self == 1 and self[1] == "0" then return "0" end
|
||||||
|
for j = #self, 1, -1 do
|
||||||
|
l[i] = self[j]
|
||||||
|
i = i + 1
|
||||||
|
end
|
||||||
|
-- trim(l)
|
||||||
|
if self.sign == -1 then
|
||||||
|
return "-"..table.concat(l)
|
||||||
|
end
|
||||||
|
return table.concat(l)
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt:inc()
|
||||||
|
-- in place
|
||||||
|
if self.sign == -1 then
|
||||||
|
udec(self)
|
||||||
|
else
|
||||||
|
uinc(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt:dec()
|
||||||
|
-- in place
|
||||||
|
if self.sign == -1 then
|
||||||
|
uinc(self)
|
||||||
|
else
|
||||||
|
udec(self)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt:toBase()
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt:fromBase()
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.divmod(a, b)
|
||||||
|
local q, r = udiv(a, b)
|
||||||
|
q.sign = a.sign * b.sign
|
||||||
|
trim(q)
|
||||||
|
trim(r)
|
||||||
|
return q, r
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt.clone(a)
|
||||||
|
local o = {unpack(a)}
|
||||||
|
o.sign = a.sign
|
||||||
|
setmetatable(o, BigInt)
|
||||||
|
return o
|
||||||
|
end
|
||||||
|
|
||||||
|
function BigInt:new(n)
|
||||||
|
local o
|
||||||
|
local t = type(n)
|
||||||
|
if t == "number" then
|
||||||
|
o = num2tab(math.floor(n))
|
||||||
|
elseif t == "string" then
|
||||||
|
o = str2tab(n)
|
||||||
|
elseif t == "table" then
|
||||||
|
if n.__tostring == self.__tostring then return n end
|
||||||
|
o = {unpack(n)}
|
||||||
|
else
|
||||||
|
o = {0, sign=1}
|
||||||
|
end
|
||||||
|
setmetatable(o, BigInt)
|
||||||
|
return o
|
||||||
|
end
|
||||||
|
|
||||||
|
setmetatable(BigInt, {__call=BigInt.new})
|
||||||
|
return BigInt
|
91
bigint_test.lua
Normal file
91
bigint_test.lua
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
B = require("bigint")
|
||||||
|
local S = string.format
|
||||||
|
|
||||||
|
function printf(...)
|
||||||
|
io.write(string.format(...))
|
||||||
|
end
|
||||||
|
|
||||||
|
function bc(exp)
|
||||||
|
local cmd = string.format('echo "%s" | bc', exp)
|
||||||
|
f = io.popen(cmd, 'r')
|
||||||
|
s = f:read('*a')
|
||||||
|
f:close()
|
||||||
|
return s:gsub("\\", ""):gsub("\n", "")
|
||||||
|
end
|
||||||
|
|
||||||
|
ops = {
|
||||||
|
['+'] = function(a, b) return a + b end,
|
||||||
|
['-'] = function(a, b) return a - b end,
|
||||||
|
['*'] = function(a, b) return a * b end,
|
||||||
|
['/'] = function(a, b) return a / b end,
|
||||||
|
['^'] = function(a, b) return a ^ b end,
|
||||||
|
}
|
||||||
|
|
||||||
|
local signs = {"", "-", "+"}
|
||||||
|
|
||||||
|
function test_op(a, b, op, msg)
|
||||||
|
local sa = tostring(a):gsub("%s+", ""):gsub("^+", "")
|
||||||
|
local sb = tostring(b):gsub("%s+", ""):gsub("^+", "")
|
||||||
|
local r = ops[op](B(a), B(b))
|
||||||
|
local sr = tostring(r)
|
||||||
|
local s = S('%s %s %s', sa, op, sb)
|
||||||
|
local s2 = msg or S('%s %s %s', a, op, b)
|
||||||
|
local br = bc(s)
|
||||||
|
assert(sr==br, S("%s: '%s' != '%s'", s2, sr, br))
|
||||||
|
print(S("[PASSED] %s", s2))
|
||||||
|
end
|
||||||
|
|
||||||
|
function test_four(a, b)
|
||||||
|
print(("-"):rep(50))
|
||||||
|
print(S("A = '%s'", a))
|
||||||
|
print(S("B = '%s'", b))
|
||||||
|
for op in pairs(ops) do
|
||||||
|
if op == '^' then goto continue end
|
||||||
|
for _, s1 in ipairs(signs) do
|
||||||
|
for _, s2 in ipairs(signs) do
|
||||||
|
local sa = s1..a
|
||||||
|
local sb = s2..b
|
||||||
|
local ma = s1.."A"
|
||||||
|
local mb = s2.."B"
|
||||||
|
test_op(sa, sb, op, S("%s %s %s", ma, op, mb))
|
||||||
|
test_op(sb, sa, op, S("%s %s %s", mb, op, ma))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
::continue::
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function test_divmod(a, b)
|
||||||
|
local q, r = B(a):divmod(B(b))
|
||||||
|
local calculatedQuotient = tostring(q)
|
||||||
|
local calculatedRemainder = tostring(r)
|
||||||
|
local sq = S("%s / %s", a, b)
|
||||||
|
local sr = S("%s %% %s", a, b)
|
||||||
|
local expectedQuotient = bc(sq)
|
||||||
|
local expectedRemainder = bc(sr)
|
||||||
|
assert(calculatedQuotient==expectedQuotient,
|
||||||
|
S("%s: '%s' != '%s'", sq, calculatedQuotient, expectedQuotient))
|
||||||
|
print(S("[PASSED] %s", sq))
|
||||||
|
assert(calculatedRemainder == expectedRemainder,
|
||||||
|
S("%s: '%s' != '%s'", sr, calculatedRemainder, expectedRemainder))
|
||||||
|
print(S("[PASSED] %s", sr))
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
test_four(
|
||||||
|
"432094820938402398402983415987430593923832340",
|
||||||
|
"234290830983245674250132908423290489320329209458645349721"
|
||||||
|
)
|
||||||
|
|
||||||
|
test_four("123", "12")
|
||||||
|
|
||||||
|
test_four(
|
||||||
|
"10000000000000000000000000000000000000000000000000000000000000000",
|
||||||
|
"99999999999999999999999999999999999999999999999999999999"
|
||||||
|
)
|
||||||
|
|
||||||
|
test_divmod("12312321", "322")
|
||||||
|
test_divmod("322", "12312321")
|
||||||
|
|
||||||
|
test_op("2", "100", "^")
|
||||||
|
test_op("5", "1024", "^")
|
Loading…
Reference in New Issue
Block a user