int i=3;
int j=4;
i ^=j ^= i ^= j;
这是个经典的写法,两个整数不用临时变量就实现了交换。
很Perfect!
但是这个在C语言中是Perfect,在C#中呢?
using System; class Test { public static void Main() { int i = 3; int j = 4; i ^= j ^= i ^= j; Console.WriteLine("I:"+i+" J:"+j); } }
但是输出为
I:0 j:3
到底发生了什么,在C里面不是运行的好好的吗.
来看看IL中间代码吧
L_0000: nop L_0001: ldc.i4.3 L_0002: stloc.0 L_0003: ldc.i4.4 L_0004: stloc.1 //前面就是初始化 L_0005: ldloc.0 L_0006: ldloc.1 L_0007: ldloc.0 L_0008: ldloc.1 L_0009: xor L_000a: dup L_000b: stloc.0//从后往前算,i ^= j算的结果存loc.0 L_000c: xor L_000d: dup L_000e: stloc.1 //loc.1的值和上面算出的loc.0的值异或,结果存loc.1 L_000f: xor L_0010: stloc.0 //问题就出在这,loc.0与上面的loc.1异或,loc.1是stloc.1算出来的,但是loc.0确实一开始就取的,所以相当于 i ^= j ^=( i ^= j); 括号部分对i值的更改没有影响到最前面的i。
所以计算结果为 3 ^= 3 , i的值就变为0了
所以代码规范告诉我们不要写这种组合表达式。
同理,i += (i+=2)也会出现C#和C的计算结果不一致的地方。