Теорема о пробном частном C++

77
17 марта 2022, 22:50

Помогите разобраться в теореме Дональда Кнута: теорема о пробном частном.

Я использую эту теорему для деления больших чисел и вычисления остатка при делении, но проблема в том, что я пересмотрел кучу сайтов, кодов, даже в его книгу смотрел чтобы понять как это все работает. И все без успешно. Не помогали даже Ctrl+C и Ctrl+V (пытался скопировать, чтобы пошагово разобраться алгоритм)

Пожалуйста, кто разбирается объясните простым языком как работает эта теорема.

Буду очень благодарен если напишите псевдокод и буду рад если закрепите все ссылочкой на простую и понятную реализацию этого алгоритма в принципе на любом языке.

Answer 1

Создадим для начала класс:

class BigInt {
    public:
    // к этим членам можно закрыть доступ
    ulong Size, SizeMax;    // Size – текущая длина
    // SizeMax – максимальная длина
    short *Coef;    // Массив коэффициентов
    // в этом случае здесь также должны быть описаны дружественные функции
    // операций над большими числами, которые будут разобраны ниже. 
    BigInt ();
    BigInt (ulong);
    BigInt (const BigInt&); 
    virtual ~BigInt ();
    void zero ();   // Обнулить число
    void update ();
    BigInt& operator = (const BigInt&);
    operator long ();   // Оператор преобразования BigInt к типу long
};

Создадим конструкторы и деструктор объектов класса:

BigInt::BigInt () {
    SizeMax = Size = 0; // Объявление вида BigInt A;
    Coef = NULL;    // Создается полностью пустое число
}
BigInt::BigInt (ulong MaxLen) {
    Coef = new short [MaxLen]; //Объявление вида BigInt A(10); 
    SizeMax = MaxLen;   // Выделяет память под MaxLen цифр Size = 0;
}
BigInt::BigInt (const BigInt &A) { // Конструктор копирования
    SizeMax = A.SizeMax;    // Создает B, равное A 
    Size = A.Size;
    Coef = new short [SizeMax];
    for (ulong i=0; i<SizeMax; i++) Coef [i] = A.Coef [i];
}
BigInt::~BigInt () {    // Деструктор (Освобождает память) 
    delete Coef;
}

Создадим операцию обнуления числа:

void BigInt::zero () {  // A.zero () – обнулить число
    for (ulong i=0; i<SizeMax; i++) Coef [i] = 0; Size = 1;
}

Создадим оператор вычисления значения числа в «обычном виде» (полезен при отладке, когда BASE=10 и числа небольшие):

BigInt::operator long () {
    long tmp = 0;   // при вычислениях может произойти переполнение
    for (ushort i=0; i<Size; i++) tmp += Coef [i] * (long) pow ( BASE, (real) i ); return tmp;
}

Создадим оператор присваивания:

inline BigInt& BigInt::operator = (const BigInt &A) { const short *a = A.Coef;
    if (this == &A) return *this;   // Если присваивание вида A=A, выйти
    if ( SizeMax < A.Size ) {   // Если размера не хватает,
        // переинициализация
        if (Coef) delete Coef;
        Coef = new short [A.Size]; SizeMax = Size = A.Size;
    } else Size = A.Size;
    for (ulong l=0; l<Size; l++) Coef [l] = a [l];
    return *this;
}

Учтены, в том числе, особые случаи применения оператора:

  1. Случай A = A;
  2. Случай необходимости выделения дополнительной памяти под коэффициенты, если размеры операндов не совпадают;
  3. Случай A = B = C (интерпретируется как A = ( B = C ) ).

Создадим оператор вывода на печать:

#define BASE_DIG 4  // Полагаем, что BASE = 10000
ostream& operator << (ostream& os, const BigInt& A) {   // Перегрузка оператора << long j, Digit=0;
    short Pow, Dec, Coef;
    os << A.Coef[A.Size-1];
    for (long i=A.Size-2; i>=0; i--) {  // Цикл вывода коэффициентов
    Pow = BASE/10;
    Coef = A.Coef[i];
        for (j=0; j<BASE_DIG; j++) {    // Цикл, выводящий каждый коэффициент
            Dec = Coef/Pow; Coef -= Dec*Pow;
            Pow /= 10;  // Очередная цифра получается делением
            os << Dec;  // коэффициента на 10j 
            Digit++;
            // Каждые 1000 цифр сопровождаются переходом строки
            if (Digit%1000==0) os << "\n\n";
            else if (Digit%50==0) os << "\t: " << Digit << "\n";
        }
    }
    return os;
}

Создадим оператор сложения:

// Вычисление C = A+B, работает вызов вида Add (A, B, A).
// Максимальный размер C должен быть достаточен для хранения суммы
void Add(const BigInt &A, const BigInt &B, BigInt &C) { ulong i;
    long temp;  // temp здесь и далее играет роль “временной” цифры,
    // до выполнения переноса. Возможно, temp > BASE.
    // Здесь и в следующих примерах
    // для быстрого доступа к коэффициентам
    // объявляются временные указатели a, b, c. 
    const short *a=A.Coef, *b=B.Coef; 
    short *c=C.Coef;
    carry = 0;  // перенос в следующий разряд
    // Добиваемся B.Size ≤ A.Size. 
    if ( A.Size < B.Size ) {
        Add(B,A,C);
        return;
    }
    // Теперь B.Size ≤ A.Size. Складываем два числа, i -номер текущей цифры
    for (i=0; i<B.Size; i++) {
        temp = a[i] + b[i] + carry;
        if (temp >= BASE) { 
            // Переполнение. Перенести единицу. 
            c[i] = temp - BASE; carry = 1; }
        else {
            c[i] = temp;    carry = 0;
        }
    } 
    // Меньшее число кончилось
    for (; i < A.Size; i++) { 
        temp = a[i] + carry; 
        if (temp >= BASE) {
            c[i] = temp - BASE;     carry = 1;  } 
        else { 
            c[i] = temp;    
            carry = 0;  }
    }
    // Если остался перенос – добавить его в дополнительный разряд
    if (carry) { 
        c[i] = carry;   
        C.Size = A.Size+1; } 
    else C.Size = A.Size;
}

Создадим оператор вычитания:

// C = A-B, должно быть A.Size >= B.Size. Работает вызов Sub(A, B, A).
// Если длины равны, но A<B: возвращается -1, результат будет дополнением. 
int Sub (const BigInt& A, const BigInt& B, BigInt& C) {
    const short *a=A.Coef, *b=B.Coef; short *c=C.Coef;
    ulong i;
    long temp, carry=0;
    if ( A.Size < B.Size ) error ("BigSub: size error");
    for (i=0; i<B.Size; i++) {
        temp = a[i] - b[i] + carry; 
        if (temp < 0) {
            c[i] = temp + BASE; carry = -1; } 
        else { c[i] = temp; carry = 0; }
    }
    for (; i<A.Size; i++) { 
        temp = a[i] + carry; 
        if (temp < 0) {
            c[i] = temp + BASE; carry = -1;  
        }
        else {
            c[i] = temp; carry = 0;
        }
    }    
    // Размер результата может быть гораздо меньше, чем у исходного числа
    // Устанавливаем его по первому положительному разряду
    i = A.Size-1;
    while ( (i>0) && (c[i]==0) ) i--; 
    C.Size = i+1;
    return carry;
}

Создадим упрощенный оператор умножения:

// C = A * B, работает вызов Smul (A, B, A).
void SMul (const BigInt &A, const short B, BigInt &C) { 
    ulong i, temp;
    const short *a=A.Coef; 
    short *c=C.Coef, carry=0; 
    for (i=0; i<A.Size;i++) {
        temp = a[i]*B + carry; carry = temp / BASE;
        c[i] = temp - carry*BASE;   // с[i] = temp % BASE ( Так очень медленно ! )
    }
    if (carry) {
        // Число удлинилось за счет переноса нового разряда
        c[i] = carry; C.Size = A.Size+1;
    }
    else C.Size = A.Size;
}

Создадим оператор умножения общего вида:

// C = A * B, работает вызов Mul (A, B, A)
void Mul (const BigInt &A, const BigInt &B, BigInt &C) { 
    ulong i, j;
    const short *a=A.Coef, *b=B.Coef; 
    short *c=C.Coef;
    ulong temp, carry;
    // Обнулить необходимую для работы часть C 
    for ( i=0; i <= A.Size + B.Size; i++ ) c[i]=0;
    // Выполнение основного цикла умножения
    for ( i = 0; i < A.Size; i++) { 
        carry = 0;
        // вычисление временного результата с одновременным прибавлением
        // его к c[i+j] (делаются переносы) 
        for (j = 0; j < B.Size; j++) {
            temp = a[i] * b[j] + c[i+j] + carry; 
            carry = temp / BASE;
            c[i+j] = temp - carry*BASE;
            }
            c[i+j] = carry;
        }
        // Установить размер по первой ненулевой цифре
        i = A.Size + B.Size - 1; 
        if ( c[i] == 0 ) i--; 
        C.Size = i+1;
}

Создадим упрощенный оператор деления:

// Частное Q = A/B. Остаток R = A%B. A, Q – длинные целые. B, R – короткие целые. 
void SDiv(const BigInt &A, const short B, BigInt &Q, short &R) {
    short r=0, *q=Q.Coef; 
    const short *a=A.Coef; 
    long i, temp;
    for ( i=A.Size-1; i>=0; i--) {  
        // идти по A, начиная от старшего разряда
        temp = r*BASE + a[i];   // r – остаток от предыдущего деления
                                // вначале r=0, temp – текущая цифра A с
                                // учетом перенесенного остатка
        q[i] = temp / B;    // i-я цифра частного
        r = temp - q[i]*B;  // остаток примет участие в вычислении
                            // следующей цифры частного
        }
    R = r;
    // Размер частного меньше, либо равен размера делимого
    i = A.Size-1;
    while ( (i>0) && (q[i]==0) ) i--; 
    Q.Size = i+1;
}

И финально, применив теорему Дональда Кнута реализуем общий оператор деления:

// Частное Q = A/B. Остаток R = A%B.
// Все операнды – длинные целые.
void Div (const BigInt &A, BigInt &B, BigInt &Q, BigInt &R) {
    // Вырожденный случай 1. Делитель больше делимого. 
    if ( A.Size < B.Size ) {
        Q.zero();
        R=A;
        return;
    }
    // Вырожденный случай 2. Делитель – короткое целое. 
    if ( B.Size == 1) {
        SDiv ( A, B.Coef[0], Q, R.Coef[0]);
        R.Size = 1; 
        return;
    }
    // Создать временный массив U, равный A
    // Максимальный размер U на цифру больше A, с учетом
    // возможного удлинения A при нормализации
    BigInt U(A.Size+1); 
    U = A; 
    U.Coef[A.Size]=0;
    // Указатели для быстрого доступа
    short *b=B.Coef, *u=U.Coef, *q=Q.Coef;
    long n=B.Size, m=U.Size-B.Size; 
    long uJ, vJ, i;
    long temp1, temp2, temp;
    short scale;    // коэффициент нормализации
    short qProbe, r;    // догадка для частного и соответствующий остаток
    short borrow, carry;    // переносы
    // Нормализация
    scale = BASE / ( b[n-1] + 1 ); 
    if (scale > 1) {
        SMul (U, scale, U);
        SMul (B, scale, B);
    }
    // Главный цикл шагов деления.
    // Каждая итерация дает очередную цифру частного.
    // vJ - текущий сдвиг B относительно U, используемый при вычитании,
    // по совместительству - индекс очередной цифры частного.
    // uJ – индекс текущей цифры U
    for (vJ = m, uJ=n+vJ; vJ>=0; --vJ, --uJ) {
        qProbe = (u[uJ]*BASE + u[uJ-1]) / b[n-1];
        r = (u[uJ]*BASE + u[uJ-1]) % b[n-1];
        // Пока не будут выполнены необходимые условия, уменьшать частное. 
        while ( r < BASE) {
            temp2 = b[n-2]*qProbe; temp1 = r*BASE + u[uJ-2];
            if ( (temp2 > temp1) || (qProbe==BASE) ) {
                // условия не выполнены, уменьшить qProbe
                // и досчитать новый остаток
                --qProbe;
                r += b[n-1];
            } else break;
        }
        // Теперь qProbe - правильное частное или на единицу больше q
        // Вычесть делитель B, умноженный на qProbe из делимого U,
        // начиная с позиции vJ+i 
        carry = 0; 
        borrow = 0; 
        short *uShift = u + vJ;
        // цикл по цифрам B 
        for (i=0; i<n;i++) {
            // получить в temp цифру произведения B*qProbe 
            temp1 = b[i]*qProbe + carry;
            carry = temp1 / BASE; temp1 -= carry*BASE;
            // Сразу же вычесть из U
            temp2 = uShift[i] - temp1 + borrow; 
            if (temp2 < 0) {
                uShift[i] = temp2 + BASE; 
                borrow = -1;             
            } else {
                uShift[i] = temp2; 
                borrow = 0;
            }
        }
        // Возможно, умноженое на qProbe число B удлинилось.
        // Если это так, то после умножения остался
        // неиспользованный перенос carry. Вычесть и его тоже. 
        temp2 = uShift[i] - carry + borrow;
        if (temp2 < 0) {
            uShift[i] = temp2 + BASE; 
            borrow = -1;         
        } else {
            uShift[i] = temp2; 
            borrow = 0;
        }
        // Прошло ли вычитание нормально ?
        if (borrow == 0) {  
            // Да, частное угадано правильно
            q[vJ] = qProbe;
        } 
        else {  
            // Нет, последний перенос при вычитании borrow = -1,
            // значит, qProbe на единицу больше истинного частного
            q[vJ] = qProbe-1;
            // добавить одно, вычтенное сверх необходимого B к U carry = 0;
            for (i=0; i<n; i++) {
                temp = uShift[i] + b[i] + carry; 
                if (temp >= BASE) {
                    uShift[i] = temp - BASE; 
                    carry = 1;
                } else {
                    uShift[i] = temp; 
                    carry = 0;
                }
            }
            uShift[i] = uShift[i] + carry - BASE;
        }
        // Обновим размер U, который после вычитания мог уменьшиться
        i = U.Size-1;
        while ( (i>0) && (u[i]==0) ) i--;
        U.Size = i+1;
    }
    // Деление завершено !
    // Размер частного равен m+1, но, возможно, первая цифра - ноль. 
    while ( (m>0) && (q[m]==0) ) m--;
    Q.Size = m+1;
    // Если происходило домножение на нормализующий множитель –
    // разделить на него. То, что осталось от U – остаток. 
    if (scale > 1) {
        short junk; // почему-то остаток junk всегда будет равен нулю… 
        SDiv ( B, scale, B, junk);
        SDiv ( U, scale, R, junk);
    } else R=U;
}

Пример словами, как работает теорема.

Дано:

Делимое - 38674

Делитель - 673

СС - 10

Первая итерация:

Длинна делителя - 3.

Берем первые 3 цифры числа - 386.

386 < 673, значит берем 3867.

Вторая итерация:

Делимое - 3867

Делитель - 673

Q.PROB = (3 * 10 + 8) / 6 = 6,....

6 * 673 = 4038, значит пробуем (Q.PROB - 1)

5 * 673 = 3365.

Остаток: 502.

Третья итерация:

Делимое - 5024

Делитель - 673

Q.PROB = (5 * 10 + 0) / 6 = 8,....

8 * 673 = 5384, значит пробуем (Q.PROB - 1)

7 * 673 = 4711.

Остаток: 313.

Больше цифр в делимом нет, значит закончили деление.

READ ALSO
Получение COUNT(*) из двух таблиц одним запросом

Получение COUNT(*) из двух таблиц одним запросом

такой вопрос уже есть, но ответа на него нет

90
Как перевести с десятичной системы счисления в римскую?

Как перевести с десятичной системы счисления в римскую?

Подскажите пожалуйста, с чего начать что б перевести с десятичной системы счисления в римскую и обратно?

85
Добавление Activity парсера HTML во Fragment

Добавление Activity парсера HTML во Fragment

Суть вопроса такая: у меня есть написанный парсер HTML в отдельном проекте, теперь я хочу впихнуть activity этого парсера во fragment 2 (вторая вкладка...

245