- •Длинная арифметика
- •Введение
- •1. Способы представления «длинных» чисел
- •2. Ввод-вывод "длинных" чисел
- •3. Сложение двух «длинных» чисел
- •4. Умножение «длинного» числа на «короткое»
- •5. Умножение «длинного» числа на «длинное»
- •6. Рекурсивный алгоритм умножения
- •7. Вычитание двух “длинных” чисел с учетом сдвига
- •8. Деление «короткого» числа на «короткое» столбиком
- •9. Деление «длинного» числа на «короткое» нацело с остатком
- •10. Сравнение двух “длинных” чисел
- •11. Задачи с решениями
- •12. Контрольные вопросы и упражнения
5. Умножение «длинного» числа на «длинное»
Для умножения двух «длинных» чисел существует несколько различных алгоритмов.
Один из них заключается в умножении каждой цифры первого множителя на каждую цифру второго и размещении каждого из таких произведений в соответствующем разряде результата. Если в каком-то разряде результата оказывается число, большее, чем максимальная цифра соответствующей системы счисления (9 в десятичной системе), то производится перенос в старший разряд. Такой алгоритм соответствует умножению двух чисел столбиком.
Приведем тексты программ, реализующих подобный алгоритм. Считывание «длинных» чисел производится из файла input.txt, а результат умножения помещается в файл output.txt. Входные данные в подобных задачах удобнее представлять в файле во избежание ошибки при вводе длинного числа с клавиатуры. Результат же по текстовому файлу проще анализировать.
Подпрограмма ReadData является примером ввода из файла «длинного» числа в случае, когда заранее неизвестна его длина. Применение в языке Turbo-Pascal логической функции SeekEoln вместо Eoln позволяет игнорировать пробелы в конце строки при вводе «длинного» числа. В языке С такую ситуацию требуется отслеживать самостоятельно.
После считывания числа из файла в первом элементе массива оказывается старшая цифра «длинного» числа. Результат же мы будем получать в массиве, первый элемент которого соответствует младшей цифре, а в нулевом элементе хранится текущая длина массива.
Туре
Abyte = array [1..10000] of byte;
Tdest = array [0..20000] of word;
Var
aLen, hLen: word;
a, b: AByte;
с: ТDest;
Procedure ReadData;
var
с:char;
begin
Assign(input,'input.txt');
Reset(input);
aLen:=0;
bLen:=0;
while not SeekEoln do {пока не достигнут конец строки}
begin
read(с);
inc(aLen);
a [aLen] :=ord(c)-ord ('0')
{перевод символа в цифру}
end;
readln;
while not SeekEoln do
begin
read(c);
inc(bLen);
b[bLen] := ord(c)-ord(' 0' )
{перевод символа в цифру}
end;
Close(input)
end;
Procedure LongXLong;
var
i, j, p, q: word;
begin
p:=aLen+bLen;
FillChar (c, 2*(p+1), 0); {заполнение массива С нулями}
for i := aLen downto 1 do
for j :=bLen downto 1 do
begin
{q — место в результате для произведения i-ой и j-ой цифры каждого из множителей}
q := p-i-j+1;
c[q] := c[q]+a[i]+b[i];
if с[q]>9 then
begin
{перенос в следующий разряд}
c[q+1] := c[q+l]+c[q] div 10;
c[q] := c[q] mod 10
end
end;
{в с[0] поместим длину результата умножения}
if с[p]<>0 then c[0] := p
else c[0] := p-l
end;
Procedure WriteResult;
var
i: word;
begin
Assign(Output,'Output.txt');
Rewrite(Output);
for i := c[0] downto i do write(c[i]);
Close(Output)
end;
{Основная программа}
Begin
ReadData;
LongXLong;
WriteResult
End.
6. Рекурсивный алгоритм умножения
Преимущество приведенных в данном параграфе программ заключается в том, что их текст является коротким и «прозрачным», а, следовательно, отладка подобной программы не составляет особого труда.
Однако существует и более эффективный рекурсивный алгоритм умножения.
Пусть х и у — два n-разрядных десятичных числа и пусть n — некоторая натуральная степень числа 2. Разобьем числа х и у на 2 равные группы, обозначив левые (старшие) цифры чисел х1 и yl соответственно, а правые (младшие) — х2 и у2. Тогда
xy = (x110n/2 + x2)(yl10n/2 + y2) =
= x1 yl10n + (x1 y2+x2yl)10n/2 + x2y2
То есть умножение двух чисел, каждое из которых имеет не более n разрядов, мы свели к трем действиям умножения чисел в два раза меньшей длины, сдвигу результатов такого умножения на соответствующее число разрядов влево и сложению длинных чисел. Однако, числа (x1+x2)и (у1+y2), вообще говоря, имеют n/2+1 разряд и поэтому их произведение нельзя вычислить непосредственным рекурсивным применением нашего алгоритма. Поэтому, чтобы сделать алгоритм универсальным, перепишем его для произвольного числа n, обозначив буквой т остаток от деления п на 2:
ху = (x110[n/2]+m + х2)(у110[n/2]+m + у2) =
= x1y110n+m + (x1y2 + x2y1)10[n/2]+m + х2у2 =
= x1y110n+m + ((x1+x2)(y1+y2)-x1y1 - х2y2)10[n/2]+m + х2у2
Теперь возможна рекурсивная реализация данного алгоритма. Если же при некотором рекурсивном вызове п окажется меньше пяти, то умножение х и у уже можно произвести непосредственно в рамках четырехбайтного целого типа.
Доказано, что с ростом п такой алгоритм является асимптотически более быстрым, нежели умножение «столбиком», однако его реализация более громоздкая и в ней приходится использовать несколько дополнительных массивов, что уменьшает максимальную длину чисел, для которых этот алгоритм может быть применен.