initial commit: import bigint

This commit is contained in:
Reza Behzadan 2020-05-13 15:37:40 +04:30
commit cf9d0892bf
4 changed files with 523 additions and 0 deletions

48
.gitignore vendored Normal file
View 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

0
README.md Normal file
View File

384
bigint.lua Normal file
View 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
View 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", "^")