What is the difference between these two macros?
#define swap(a, b) (((a) ^ (b)) && ((a) ^= (b) ^= (a) ^= (b)))
Or
#define swap(a, b) (((a) ^ (b)) && ((b) ^= (a) ^= (b), (a) ^= (b)))
I saw the second macro here but couldn't understand why it wasn't written like the first one? Is there a special reason that I missed?
First will invoke undefined behavior in both C99 and C11.
In C99, it can be understood as; they will invoke undefined behavior because of the lack of the sequence points.
C-faq:
Explanation:
The first one is modifying
a
twice between two sequence points and hence behavior is undefined as per statement: Between the previous and next sequence point an object shall have its stored value modified at most once by the evaluation of an expression. That's it (no need to think aboutb
).C11 documentation says:
6.5 Expressions (p2):
In
(a) ^= (b) ^= (a) ^= (b)
, side effect ona
is unsequenced and hence invoke undefined behavior. It should be noted that C11 6.5 p1 says that:This guarantees that in
all subexpressions 1, 2, 3 and 4 are guaranteed to be computed before the result computation of left most
^=
operator. But, this doesn't guarantee that side effect of expression 3 is guaranteed before the value computation of the result of the left most^=
operator.1. Emphasis is mine.