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的计算结果不一致的地方。