2022-09-15 08:40:48 +03:00
|
|
|
|
#include <iostream>
|
|
|
|
|
#include <iomanip>
|
|
|
|
|
#include <cstring>
|
|
|
|
|
#include "number.h"
|
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
Number::Number(int a)
|
|
|
|
|
{
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_CONSTRUCTOR
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "(Number constructor " << a << " -> ";
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
// Находим размер необходимой памяти под это число
|
|
|
|
|
int temp = a;
|
|
|
|
|
capacity = 0;
|
|
|
|
|
while (temp != 0) {
|
|
|
|
|
temp /= base;
|
|
|
|
|
capacity += 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Отдельно обрабатываем случай, когда число равно 0
|
|
|
|
|
if (capacity == 0)
|
|
|
|
|
capacity = 1;
|
|
|
|
|
|
|
|
|
|
// Выделяем память и записывем число a в массив data
|
|
|
|
|
// Например, число 12345678 представится в виде массива [78, 56, 34, 12]
|
|
|
|
|
|
|
|
|
|
data = new char[capacity];
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < capacity; ++i) {
|
|
|
|
|
data[i] = a % base;
|
|
|
|
|
a /= base;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// В данном случае размер будет равен вместимости
|
|
|
|
|
size = capacity;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_CONSTRUCTOR
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << *this << ")" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-09-15 08:40:48 +03:00
|
|
|
|
// Конструктор по умолчанию
|
2022-09-15 10:29:14 +03:00
|
|
|
|
Number::Number():Number(0){}
|
|
|
|
|
|
2022-09-15 08:40:48 +03:00
|
|
|
|
// Конструктор копирования
|
2022-09-15 10:29:14 +03:00
|
|
|
|
Number::Number(const Number & n)
|
|
|
|
|
{
|
|
|
|
|
size = n.size;
|
|
|
|
|
capacity = n.capacity;
|
|
|
|
|
data = new char[capacity];
|
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
|
data[i] = n.data[i];
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Number::Number(const char *str)
|
|
|
|
|
{
|
|
|
|
|
int len = std::strlen(str);
|
|
|
|
|
size = (len + len % 2) / 2;
|
|
|
|
|
capacity = size;
|
|
|
|
|
data = new char[capacity];
|
|
|
|
|
char buf[2];
|
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
|
buf[1] = str[len - 2 * i - 1];
|
|
|
|
|
if (len - 2 * i - 1 > 0) {
|
|
|
|
|
buf[0] = str[len - 2 * i - 2];
|
|
|
|
|
} else {
|
|
|
|
|
buf[0] = '0';
|
|
|
|
|
}
|
|
|
|
|
data[i] = std::stoi(buf);
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Number::~Number()
|
|
|
|
|
{
|
|
|
|
|
delete[]data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Number & Number::operator=(const Number & right)
|
|
|
|
|
{
|
|
|
|
|
capacity = right.capacity;
|
|
|
|
|
size = right.size;
|
|
|
|
|
data = new char[capacity];
|
|
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
|
data[i] = right.data[i];
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return *this;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
Number Number::operator+(Number a)
|
|
|
|
|
{
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_ADD
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "arg1=" << a << "capacity=" << a.
|
|
|
|
|
capacity << ",size=" << a.size << std::endl;
|
|
|
|
|
std::cout << "arg2=" << *this << "capacity=" << this->
|
|
|
|
|
capacity << ",size=" << this->size << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
Number result;
|
|
|
|
|
Number temp;
|
|
|
|
|
int i;
|
|
|
|
|
int carry = 0;
|
|
|
|
|
if (size < a.size) {
|
|
|
|
|
temp = *this;
|
|
|
|
|
*this = a;
|
|
|
|
|
a = temp;
|
|
|
|
|
}
|
|
|
|
|
result.capacity = size + 1;
|
|
|
|
|
//result.data = new char[capacity];
|
|
|
|
|
result.data = (char *) calloc(result.capacity, sizeof(char));
|
|
|
|
|
for (i = 0; i < a.size; ++i) {
|
|
|
|
|
result.data[i] = (data[i] + a.data[i] + carry) % base;
|
|
|
|
|
carry = (data[i] + a.data[i] + carry) / base;
|
|
|
|
|
}
|
|
|
|
|
for (; i < size; ++i) {
|
|
|
|
|
result.data[i] = (data[i] + carry) % base;
|
|
|
|
|
carry = (data[i] + carry) / base;
|
|
|
|
|
}
|
|
|
|
|
if (carry) {
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_ADD
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "applied carry" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
result.data[i] = carry;
|
|
|
|
|
result.size = size + 1;
|
|
|
|
|
} else {
|
|
|
|
|
result.size = size;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_ADD
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << result << " capacity=" << result.
|
|
|
|
|
capacity << ",size=" << result.size << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return result;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
void Number::operator+=(const Number & a)
|
|
|
|
|
{
|
|
|
|
|
*this = *this + a;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Number::isEven() const
|
|
|
|
|
{
|
|
|
|
|
if (data[0] % 2) {
|
|
|
|
|
return false;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Number Number::operator*(const Number & right) const
|
|
|
|
|
{
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_MUL
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::
|
|
|
|
|
cout << "arg1=" << *this << "(capacity=" << capacity << ",size=" <<
|
|
|
|
|
size << ")" << " " << "arg2=" << right << "(capacity=" << right.
|
|
|
|
|
capacity << ",size=" << right.size << ")" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
int i, j;
|
|
|
|
|
int temp;
|
|
|
|
|
Number result;
|
|
|
|
|
|
|
|
|
|
result.capacity = capacity + right.capacity;
|
|
|
|
|
int *carry = (int *) std::calloc(result.capacity, sizeof(int));
|
|
|
|
|
|
|
|
|
|
result.data = (char *) calloc(result.capacity, sizeof(char));
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_MUL
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "carry:[" << carry[0];
|
|
|
|
|
for (int k = 1; k < result.capacity; ++k) {
|
|
|
|
|
std::cout << "," << carry[k];
|
|
|
|
|
}
|
|
|
|
|
std::cout << "]" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
for (i = 0; i < size; ++i) {
|
|
|
|
|
for (j = 0; j < right.size; ++j) {
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_MUL
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << i + j << ":" << static_cast <
|
|
|
|
|
int >(result.data[i + j]) << " + " << static_cast <
|
|
|
|
|
int >(data[i]) << " * " << static_cast <
|
|
|
|
|
int >(right.data[j]) << " + " << carry[i + j] << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
temp =
|
|
|
|
|
(result.data[i + j] + data[i] * right.data[j] +
|
|
|
|
|
carry[i + j]);
|
|
|
|
|
result.data[i + j] = temp % base;
|
|
|
|
|
carry[i + j + 1] += temp / base;
|
|
|
|
|
carry[i + j] = 0;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_MUL
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "result before applying carry:" << result << std::endl;
|
|
|
|
|
std::cout << "carry:[" << carry[0];
|
|
|
|
|
for (int k = 1; k < result.capacity; ++k) {
|
|
|
|
|
std::cout << "," << carry[k];
|
|
|
|
|
}
|
|
|
|
|
std::cout << "]" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
if (carry[i + j - 1]) {
|
|
|
|
|
result.data[i + j - 1] = carry[i + j - 1];
|
|
|
|
|
result.size = i + j;
|
|
|
|
|
} else {
|
|
|
|
|
result.size = i + j - 1;
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_MUL
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "before correcting capacity, result=" << result << std::
|
|
|
|
|
endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
// correcting capacity
|
|
|
|
|
/*char* temp_data = (char *)calloc(result.size, sizeof(char));
|
|
|
|
|
for (i = 0; i < result.size; ++i) {
|
|
|
|
|
temp_data[i] = result.data[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(result.data);
|
|
|
|
|
result.capacity = result.size;
|
|
|
|
|
result.data = (char*)calloc(result.size,sizeof(char));
|
|
|
|
|
for (i = 0; i < result.size; ++i) {
|
|
|
|
|
result.data[i] = temp_data[i];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free(temp_data); */
|
|
|
|
|
free(carry);
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_MUL
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "return value=" << result << "(capacity=" << result.
|
|
|
|
|
capacity << ",size=" << result.
|
|
|
|
|
size << ")" << std::endl << "======" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return result;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
void Number::operator*=(const Number & a)
|
|
|
|
|
{
|
|
|
|
|
*this = *this * a;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Number::operator==(const Number & a) const
|
|
|
|
|
{
|
|
|
|
|
if (size != a.size) {
|
|
|
|
|
return false;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
2022-09-15 10:29:14 +03:00
|
|
|
|
for (int i = 0; i < size; ++i) {
|
|
|
|
|
if (data[i] != a.data[i]) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Number::operator!=(const Number & a) const
|
|
|
|
|
{
|
|
|
|
|
return not(*this == a);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Number::operator>(const Number & a) const
|
|
|
|
|
{
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_COMP
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::
|
|
|
|
|
cout << "comp " << *this << "(size=" << size << ") and " << a <<
|
|
|
|
|
"(size=" << a.size << ")" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
if (size > a.size) {
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_COMP
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "size > a.size => true" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
if (size < a.size) {
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_COMP
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "size < a.size => false" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
for (int i = size - 1; i >= 0; --i) {
|
|
|
|
|
if (data[i] > a.data[i]) {
|
|
|
|
|
return true;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_COMP
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << static_cast <
|
|
|
|
|
int >(data[i]) << ">" << static_cast <
|
|
|
|
|
int >(a.data[i]) << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
|
|
|
|
if (data[i] < a.data[i]) {
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_COMP
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << static_cast <
|
|
|
|
|
int >(data[i]) << "<" << static_cast <
|
|
|
|
|
int >(a.data[i]) << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_COMP
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "using final false" << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Number::operator<(const Number & a) const
|
|
|
|
|
{
|
|
|
|
|
return not(*this > a) and(*this != a);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Number::div2()
|
|
|
|
|
{
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_DIV2
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "n = " << *this << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
int carry = 0;
|
|
|
|
|
int temp;
|
|
|
|
|
for (int i = size - 1; i >= 0; --i) {
|
|
|
|
|
temp = data[i] + carry * base;
|
|
|
|
|
data[i] = temp / 2;
|
|
|
|
|
carry = temp % 2;
|
|
|
|
|
}
|
|
|
|
|
if (data[size - 1] == 0) {
|
|
|
|
|
--size;
|
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#ifdef _DEBUG_DIV2
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::cout << "unstripped result " << *this << std::endl;
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
2022-09-15 10:29:14 +03:00
|
|
|
|
}
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
|
|
|
|
|
2022-09-15 10:29:14 +03:00
|
|
|
|
std::ostream & operator<<(std::ostream & stream, const Number & right)
|
2022-09-15 08:40:48 +03:00
|
|
|
|
{
|
|
|
|
|
#ifdef _DEBUG_COUT
|
|
|
|
|
stream << "[";
|
|
|
|
|
for (std::size_t i = 0; i < right.size; ++i) {
|
2022-09-15 10:29:14 +03:00
|
|
|
|
stream << static_cast <
|
|
|
|
|
int >(right.data[right.size - 2 - i]) << ",";
|
2022-09-15 08:40:48 +03:00
|
|
|
|
}
|
|
|
|
|
stream << "]";
|
|
|
|
|
#else
|
2022-09-15 10:29:14 +03:00
|
|
|
|
// Печатаем самый большой разряд
|
|
|
|
|
stream << (int) right.data[right.size - 1];
|
2022-09-15 08:40:48 +03:00
|
|
|
|
|
|
|
|
|
// Печатаем остальные разряды с заполнением нулями до 2-х цифр
|
|
|
|
|
// setfill и setw это то же самое, что и в языке C спецификатор %02d
|
|
|
|
|
for (std::size_t i = 0; i < right.size - 1; ++i)
|
2022-09-15 10:29:14 +03:00
|
|
|
|
stream << std::setfill('0') << std::setw(2) << (int) right.
|
|
|
|
|
data[right.size - 2 - i];
|
2022-09-15 08:40:48 +03:00
|
|
|
|
#endif
|
|
|
|
|
return stream;
|
|
|
|
|
}
|