I recently came across a post What is the correct answer for cout << c++ << c;? and was wondering whether the output of
int c = 0;
printf ("%d %d", c++, c);
is also undefined??
I have studied in lectures that post-fix and prefix operators increment value only after getting a semicolon. So according to me, the output 0 0 is correct !!!
Send your lecturer to me so that I can
take a baseball bat to himpolitely point out his mistake.Exactly when the side effect of either pre- or postfix
++and--is applied is unspecified, apart from the requirement that it happen before the next sequence point. In an expression likeamay be updated immediately aftera++has been evaluated, or the update may be deferred untila++ * bhas been evaluated and the result assigned tox, or anywhere in between.This is why expressions like
i++ * i++andprintf("%d %d", c++, c)anda[i++] = iand a host of others are all bad juju. You will get different results based on the compiler, optimization settings, surrounding code, etc. The language standard explicitly leaves the behavior undefined so that the compiler is under no obligation to “do the right thing”, whatever the right thing may be. Remember, the definition for undefined behavior isThis is a deliberate design decision – the rationale for leaving the order of these operations unspecified is to give the implementation freedom to rearrange the evaluation order for optimization purposes. However, in exchange for this freedom, certain operations will not have well-defined results.
Note that a compiler is free to try to detect these cases and issue a diagnostic;
printf("%d %d", c++, c);would be easy enough to catch, but this would be a bugger to detect in the general case. Imagine if that had been writtenprintf("%d %d", (*p)++, c); ifppoints toc, then the behavior is undefined, otherwise it’s okay. Ifpis assigned in a different translation unit, then there’s no way to know at compile time whether this is a problem or not.This concept is not difficult to understand, yet it is one of the most consistently misunderstood (and mis-taught) aspects of the C language. No doubt this is why the Java and C# language specifications force a specific evaluation order for everything (all operands are evaluated left-to-right, and all side effects are applied immediately).