Здравствуйте, Rosigma, Вы писали:
АТ>>С практической точки зрения, если забыть про неопределенное поведение, никто не гарантирует, что постинкретмент 'x' будет произведен до присваивания 'x' нового значения. Если постинкремент будет произведен после присвивания, то финальное знечение 'x' будет '1'. Если до — то '0'. Финальный 'printf', может напечатать как '0', так и '1'.
R>Если постинкремент реализован посредством преинкремента,
???
R>то постинкретмент 'x' будет произведен до присваивания 'x' нового значения. И результат должен быть всегда 0 и никогда 1.
Нет. Еще раз: инкремент (пост- или пре- — неважно) может быть произведен как до так и после присваивания. Это зависит от большого количества труднопредсказуемых факторов, типа установок компиляции, контекста, версии компилятора и дня недели.
Здравствуйте, Андрей Тарасевич, Вы писали:
АТ>Нет. Еще раз: инкремент (пост- или пре- — неважно) может быть произведен как до так и после присваивания. Это зависит от большого количества труднопредсказуемых факторов, типа установок компиляции, контекста, версии компилятора и дня недели.
R>>Вот именно так и будет.
АТ>Нет, как "будет" предказать заранее невозможно. На то оно и неопределенное поведение.
Операция инкремента имеет высший приоритет, чем операция присваивания. Поэтому она должна выполняться перед операцией присваивания. И никакого здесь неопределенного поведения.
R>>>Вот именно так и будет.
АТ>>Нет, как "будет" предказать заранее невозможно. На то оно и неопределенное поведение.
R>Операция инкремента имеет высший приоритет, чем операция присваивания. Поэтому она должна выполняться перед операцией присваивания.
Во-первых, приоритеты операций не имеют никакого отношения к тому, в каком порядке они будут выполняться. Приоритеты операций задают только группировку операций в выражении.
Во-вторых, "выполнение" — несколько неконкретный термин. Операторы в выражениях в С/С++ вычисляются и при этом могут обладать еще побочными эффектами.
Вычисление оператора — 'i++' заключается в получении старого значения 'i'. И все. Сам же инкремент 'i' не является частью процесса вычисления 'i++'. Это побочный эффект 'i++'. Побочный эффект может произойти (т.е. 'i' может увеличиться на 1) в любой момент до конца вычисления данного выражения.
Вычисление выражения вида 'i = v[i]' заключается в получении значения 'v[i]' и приведении его к типу переменной 'i' (пусть 'int'). Смена значения 'i' присваиванием является побочным эффектом данного выражения и может произойти в любой момент до конца вычисления данного выражения.
Таким образом, в данном случае выполнение нашего выражения 'i = v[i++]' эквивалентно вычислению значения 'V' равного '(int) v[I]' (где 'I' — значение переменной 'i' до инкремента) и выполнению двух побочных эффектов: "занести значение 'I + 1' в переменную 'i'" и "занести значение 'A' в переменную 'i'". Языки С/С++ не дают абсолютно никаких гарантий по поводу того, в каком порядке произойдут эти побочные эффекты. Отсюда и получаются два ранее приводившихся варианта. Приоритеты операций же не имеют к этому никакого отношения.
R>И никакого здесь неопределенного поведения.
Неопределенное поведение в данном случае оговороено черным по белому в стандартах языков С и С++.
АТ>Таким образом, в данном случае выполнение нашего выражения 'i = v[i++]' эквивалентно вычислению значения 'V' равного '(int) v[I]' (где 'I' — значение переменной 'i' до инкремента) и выполнению двух побочных эффектов: "занести значение 'I + 1' в переменную 'i'" и "занести значение 'A' в переменную 'i'".
Здесь опечатка. Читать: "занести значение 'V' в переменную 'i'"
Здравствуйте, _wind_, Вы писали:
__>Здравствуйте, _wind_, Вы писали:
__>>Здравствуйте, Аноним, Вы писали:
А>>>Именно такой код: А>>>
А>>>#include <stdio.h>
А>>>void main()
А>>>{
А>>> int x = 0;
А>>> if (x = 0 || x++)
А>>> printf("%d\n", x);
А>>> printf("%d\n", x);
А>>>}
А>>>
А>>>что выдаст и почему?
А>>>
__>>1
__>проверил. Выдаёт 0. __>почему x++ не увеличивает значение икса? __>Если первый операнд оператора || есть false, то должен выполниться и второй опреанд!
в действительности он обрабатывается и он был бы "1" если бы симолы "++" стояли бы перед иксом, а так, он выполняется после определения истинности условия..... таким образо в любом случаи должна быть выведена на экран "1".
Здравствуйте, greenya, Вы писали:
G>Здравствуйте, _wind_, Вы писали:
__>>Здравствуйте, _wind_, Вы писали:
__>>>Здравствуйте, Аноним, Вы писали:
А>>>>Именно такой код: А>>>>
А>>>>#include <stdio.h>
А>>>>void main()
А>>>>{
А>>>> int x = 0;
А>>>> if (x = 0 || x++)
А>>>> printf("%d\n", x);
А>>>> printf("%d\n", x);
А>>>>}
А>>>>
А>>>>что выдаст и почему?
А>>>>
__>>>1
__>>проверил. Выдаёт 0. __>>почему x++ не увеличивает значение икса? __>>Если первый операнд оператора || есть false, то должен выполниться и второй опреанд!
G>в действительности он обрабатывается и он был бы "1" если бы симолы "++" стояли бы перед иксом, а так, он выполняется после определения истинности условия..... таким образо в любом случаи должна быть выведена на экран "1".
Здравствуйте, Rosigma, Вы писали:
R>Здравствуйте, greenya, Вы писали:
G>>Здравствуйте, _wind_, Вы писали:
__>>>Здравствуйте, _wind_, Вы писали:
__>>>>Здравствуйте, Аноним, Вы писали:
А>>>>>Именно такой код: А>>>>>
А>>>>>#include <stdio.h>
А>>>>>void main()
А>>>>>{
А>>>>> int x = 0;
А>>>>> if (x = 0 || x++)
А>>>>> printf("%d\n", x);
А>>>>> printf("%d\n", x);
А>>>>>}
А>>>>>
А>>>>>что выдаст и почему?
А>>>>>
__>>>>1
__>>>проверил. Выдаёт 0. __>>>почему x++ не увеличивает значение икса? __>>>Если первый операнд оператора || есть false, то должен выполниться и второй опреанд!
G>>в действительности он обрабатывается и он был бы "1" если бы симолы "++" стояли бы перед иксом, а так, он выполняется после определения истинности условия..... таким образо в любом случаи должна быть выведена на экран "1".
R>У кого компилятор выдает "1"???!