На данный вопрос уже ответили:
i = 5;
i = ++i + ++i
Как компилятор считает? Язык C++
Результат вычисления
i = ++i + ++i
неопределен и может быть как 13, так и 14. Причиной неопределенного поведения в данном выражении является правило, по которому один и тот же объект между двумя точками следования не может быть изменен более одного раза. В выражении
i = ++i + ++i
операция + не является точкой следования и переменная i инкрементируется два раза. Компилятор по старшинству операций обязан сначала вычислить значения левого и правого операндов и только потом сложить их. Но при этом то, как он будет вычислять операнды неопределено. Поэтому компилятор может сначала инкрементировать переменную первый раз, потом инкрементировать переменную второй раз, затем подставить значение переменной в обе части сложения и получить 14. Но компилятор волен так же сначала инкрементировать переменную слева и подставить ее как первый операнд, затем инкрементировать переменную справа и подставить ее как второй операнд,потом сложить оба результата и получить 13.
Как правильно заметил ixSci в с++11 правило точки следования было заменено на правило отношения расположено-перед. Но смысл тот же: многократное изменение объекта в сочетании с чтением этого объекта - не упорядоченно.
Результат вычисления выражения ++i + ++i
, как уже верно написали, может быть любым, т.к. это классический пример неопределенного поведения. Формально, компилятор имеет право выдать результат 42 с тем же успехом, как 12, 13 или 14. Может сформатировать диск или запустить ракеты.
Еще одно объяснение: a + b
эквивалентно вызову функции operator+(a, b)
(на самом деле не совсем, т.к. это верно не для любых операторов, об этом ниже) а порядок вычисления операндов функции не определен, это явно прописано в стандарте. Однако есть три бинарных оператора, порядок вычисления аргументов которых определен: это операторы ,
(запятая), &&
и ||
(логические И и ИЛИ). Их операнды всегда вычисляются слева направо. То есть выражение
bool result = ++i && ++i;
не содержит UB (если i имеет встроенный тип). Также интересно, что если Вы перегружаете операторы ,
, &&
и ||
для пользовательского класса, это поведение меняется, и порядок вычисления операндов становится неопределенным. Это причина того, что без крайней нужды эти операторы перегружать не рекомендуется.
Ответ может быть любым. Это классическая задачка на неопределенное поведение компилятора. Тут переменная изменяется 2 раза между точками следования
Виртуальный выделенный сервер (VDS) становится отличным выбором
Здравствуйте, проблема с компиляцией программы на OS Linux Ubuntu, ОС стоит на плате odroid-xu4, с ARM процессоромБиблиотеку подключил, ругается только...
Есть класс для работы с оборудованием, который имеет 2 основных метода
Хочу поставить иконку для приложения, следуя инструкции с http://docqt
С числами всё понятно, там младший разряд можно получить с помощью деления по модулю 10: 2498 % 10 = 8