Почему я не могу разделить большое число на отрицательное число C++

Там нет реальной необходимости решения этого, я просто хочу знать, почему. Давайте возьмем два числа:

#include <iostream>
using namespace std;

int main()
{
    unsigned long long int a = 17446744073709551615;
    signed long long int b = -30000000003;
    signed int c;
    c = a/b;
    cout << "\n\n\n" << c << endl;
}

Теперь в последнее время я получаю нулевой ответ. Размер моего длинного длинного составляет 8 байтов, так что более чем достаточно, чтобы взять его с беззнаковой меткой. Переменная C также должна быть достаточно большой, чтобы обрабатывать ответ. (Это должно быть -581 558 136, согласно Google). Так...

Изменить Я хотел бы отметить, что на моей машине...

Использование numeric_limits a хорошо ложится в пределах максимума 18446744073709551615, а значение b попадает в минимальные пределы -9223372036854775808.

2 ответа

Решение

У вас происходит ряд неявных преобразований, большинство из которых не нужны.

unsigned long long int a = 17446744073709551615;

Десятичный целочисленный литерал без суффикса имеет тип int, long int, или же long long int; это никогда не беззнаковый тип. Это конкретное значение почти наверняка превышает максимальное значение long long int (263-1). Если ваш компилятор не имеет целочисленный тип со знаком шире, чем 64 бита, это делает вашу программу плохо сформированной.

Добавить ULL суффикс, чтобы гарантировать, что литерал имеет правильный тип:

unsigned long long int a = 17446744073709551615ULL;

Значение оказывается между 263-1 и 264-1, поэтому оно подходит для 64-разрядного типа без знака, но не для 64-разрядного типа со знаком.

(На самом деле только U было бы достаточно, но это не помешает быть явным.)

signed long long int b = -30000000003;

Это не должно быть проблемой. 30000000003 имеет некоторый целочисленный тип со знаком; если ваш компилятор поддерживает long longшириной не менее 64 бит, переполнения нет. Тем не менее, пока вам нужен суффикс на значении a, не мешало бы быть явным:

signed long long int b = -30000000003LL;

Теперь у нас есть:

signed int c;
c = a/b;

Разделение unsigned long long по signed long long вызывает преобразование подписанного операнда в unsigned long long, В этом случае конвертируемое значение является отрицательным, поэтому оно преобразуется в большое положительное значение. преобразование -30000000003 в unsigned long long доходность 18446744043709551613, Разделив 17446744073709551615 от 18446744043709551613 дает ноль.

Если ваш компилятор не поддерживает целые числа шире, чем 64 бита (большинство из них не поддерживают), вы не сможете напрямую делить 17446744073709551615 от -30000000003 и получить математически правильный ответ, так как нет целочисленного типа, который может представлять оба значения. Все арифметические операторы (кроме операторов сдвига) требуют операнды одного типа с неявными преобразованиями, применяемыми по мере необходимости.

В этом конкретном случае вы можете разделить 17446744073709551615ULL от 30000000003ULL а затем учитывать знак. (Проверьте правила языка для деления отрицательных целых чисел.)

Если вам действительно нужно сделать это в целом, вы можете прибегнуть к плавающей запятой (что означает, что вы, вероятно, потеряете некоторую точность) или использовать некоторый целочисленный арифметический пакет произвольной ширины, такой как GMP.

b рассматривается как число без знака, которое больше, чем a. Следовательно, вы получаете ответ как 0.

Попробуйте использовать это как

c = abs(a) / abs (b)
if ((a < 0 && b > 0 ) || (a> 0 && b < 0))
   return -c;
return c;
Другие вопросы по тегам