-
Notifications
You must be signed in to change notification settings - Fork 39
/
Copy pathkaratsuba.py
66 lines (52 loc) · 1.6 KB
/
karatsuba.py
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
#!/usr/bin/python
# Date: 2017-12-17
#
# Description:
# Karatsuba algo an efficiant way to multiply very large numbers. Conventional
# way of multiplication has complexity of O(n^2) as every bit of one number is
# multiplied by every other. Karatsuba is efficient than quadratic.
#
# Approach:
# Both numbers can be expressed as high and low components like
# x = (10^n/2)*x_H + x_L and y = (10^n/2)*y_H + y_L
# where n is max of number of digits in x or y.
#
# So, x*y = (10^n)*x_H*y_H + (10^n/2)*(x_H*y_L + x_L*y_H) + x_L*y_L
# above expression requires 4 multiplies to get the result which can be reduced
# to 3 as we can find x_H*y_L + x_L*y_H using a hack:
#
# x_H*y_L + x_L*y_H = (x_H + x_L)*(y_H + y_L) - x_H*y_H - x_L*y_L
# and we have already found x_H*y_H and x_L*y_L. so we are done in 3 multiply
# operations instead of 4.
#
# Reference:
# https://brilliant.org/wiki/karatsuba-algorithm/
#
# Complexity:
# O(n^1.58), better than conventional algo.
def karatsuba(x, y):
"""
Multiply 2 numbers by karatsuba algo.
Args:
x: first number to be multiplied.
y: second number to be multiplied.
Returns:
Product of x and y.
"""
# base case
if (x < 10 and y < 10):
return x*y
n = max(len(str(x)), len(str(y)))
m = n/2
div_factor = 10**m
x_H = x / div_factor
x_L = x % div_factor
y_H = y / div_factor
y_L = y % div_factor
a = karatsuba(x_H, y_H)
d = karatsuba(x_L, y_L)
e = karatsuba(x_H + x_L, y_H + y_L) - a - d
return (a*(10**(m*2)) + e*(10**m) + d)
num_1 = input("Enter first number: ")
num_2 = input("Enter second number: ")
print(karatsuba(int(num_1), int(num_2)))