gpt4 book ai didi

python - Karatsuba 算法适用于小数但不适用于大数,不明白为什么

转载 作者:塔克拉玛干 更新时间:2023-11-03 04:58:33 25 4
gpt4 key购买 nike

我对编程还比较陌生,我不希望使用这种算法在运行时间方面特别高效,而只是尝试复制 Karatsuba 算法并使其发挥作用。

我试过很多数字和小数字(比如 y = 40004009343254,x = 40004001343234) 工作正常,当数字大小增加时(如 y = 4000400934325423423,x = 4000400134323432423),算法停止正常工作并返回相似但不正确的答案。

如果能提供任何可能出错的线索,我们将不胜感激!

注意:这个线程不是关于效率而是关于获得正确的结果。也就是说,有关效率的评论也将被考虑在内并受到赞赏。

代码:

y = 4000400934325423423
x = 4000400134323432423

def firsthalf(array):
firsthalf = array[:len(array)/2]
return firsthalf
def secondhalf(array):
secondhalf = array[len(array)/2:]
return secondhalf
def arrayjoint(array):
jointarray = long(''.join(map(str,array)))
return jointarray
def karatsuba(x,y):
if len(str(x)) == 0 or len(str(y)) == 0:
return "Can't multiply by a NULL value!"
if x < 10 or y < 10:
return x * y
x_array = [long(i) for i in str(x)]
y_array = [long(i) for i in str(y)]
firsthalf_xarray = firsthalf(x_array)
secondhalf_xarray = secondhalf(x_array)
firsthalf_yarray = firsthalf(y_array)
secondhalf_yarray = secondhalf(y_array)
half_size = max(len(secondhalf_yarray), len(secondhalf_xarray))
firsthalf_x = arrayjoint(firsthalf_xarray)
secondhalf_x = arrayjoint(secondhalf_xarray)
firsthalf_y = arrayjoint(firsthalf_yarray)
secondhalf_y = arrayjoint(secondhalf_yarray)
sum_x = firsthalf_x + secondhalf_x
sum_y = firsthalf_y + secondhalf_y
first = karatsuba(firsthalf_x,firsthalf_y)
second = karatsuba(sum_x, sum_y)
third = karatsuba(secondhalf_x,secondhalf_y)
return first * 10 ** (2 * half_size) + ((second - first - third) * (10 ** half_size)) + third

result = karatsuba(x,y)
result_correct = x*y
result = str(result)
result_correct = str(result_correct)
file = open("result.txt", "w")
file.write(str(result) + "\n" + str(result_correct))
file.close

最佳答案

这不是 float 的问题,因为 Python 有大数。

问题在于,当输入的长度不同时,您会将它们拆分到不同的位置,这会破坏 Karatsuba 算法的基础代数。通过在索引 -half_size 处拆分(即,后半部分有 half_size 数字),我们确保 10**half_size 是正确的基数。试试这个:

def digits_to_long(x_array):
return long(''.join(x_array)) if x_array else 0L


def karatsuba(x, y):
if x < 10 or y < 10:
return x * y
x_array = str(x)
y_array = str(y)
half_size = max(len(x_array), len(y_array)) // 2
firsthalf_x = digits_to_long(x_array[:-half_size])
secondhalf_x = digits_to_long(x_array[-half_size:])
firsthalf_y = digits_to_long(y_array[:-half_size])
secondhalf_y = digits_to_long(y_array[-half_size:])
sum_x = firsthalf_x + secondhalf_x
sum_y = firsthalf_y + secondhalf_y
first = karatsuba(firsthalf_x, firsthalf_y)
second = karatsuba(sum_x, sum_y)
third = karatsuba(secondhalf_x, secondhalf_y)
return first * 10**(2 * half_size) + (
(second - first - third) * (10**half_size)) + third


import random
for i in range(10000):
x = random.randrange(10**18)
y = random.randrange(10**18)
assert karatsuba(x, y) == x * y

关于python - Karatsuba 算法适用于小数但不适用于大数,不明白为什么,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41063543/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com