4个按位逻辑运算符都用于整型数据,包括char。之所以叫作按位(bitwise)运算,是因为这些操作都是针对每一个位进行,不影响它左右两边的位。不要把这些运算符与常规的逻辑运算符(&&、||和!)混淆,常规的逻辑运算符操作的是整个值。
1.二进制反码或按位取反:~一元运算符~ 把1变为0,把0变为1。如下例子所示:
~(10011010) // expression
(01100101) // resulting value
假设val的类型是unsigned char,已被赋值为2。在二进制中,00000010表示2。那么,~val的值是11111101,即253。注意,该运算符不会改变val的值,就像3 * val不会改变val的值一样,val仍然是2。但是,该运算符确实创建了一个可以使用或赋值的新值:
newval = ~val;
printf("%d", ~val);
如果要把val的值改为~val,使用下面这条语句:
val = ~val;
2.按位与:&
二元运算符&通过逐位比较两个运算对象,生成一个新值。对于每个位,只有两个运算对象中相应的位都为1时,结果才为1(从真/假方面看,只有当两个位都为真时,结果才为真)。因此,对下面的表达式求值:
(10010011) & (00111101) // expression
evaluates to the following value:
(00010001) // resulting value
C有一个按位与和赋值结合的运算符:&=。下面两条语句产生的最终结果相同:
val &= 0377;
val = val & 0377;
3.按位或:|
二元运算符|,通过逐位比较两个运算对象,生成一个新值。对于每个位,如果两个运算对象中相应的位为1,结果就为1(从真/假方面看,如果两个运算对象中相应的一个位为真或两个位都为真,那么结果为真)。因此,对下面的表达式求值:
(10010011) | (00111101) // expression
除了编号为6的位,这两个运算对象的其他位至少有一个位为1,得:
(10111111) // resulting value
C有一个按位或和赋值结合的运算符:|=。下面两条语句产生的最终作用相同:
val |= 0377;
val = val | 0377;
4.按位异或:^
二元运算符逐位比较两个运算对象。对于每个位,如果两个运算对象中相应的位一个为1(但不是两个为1),结果为1(从真/假方面看,如果两个运算对象中相应的一个位为真且不是两个为同为1,那么结果为真)。因此,对下面表达式求值:
(10010011) ^ (00111101) // expression
编号为0的位都是1,所以结果为0,得:
(10101110) // resulting value
C有一个按位异或和赋值结合的运算符:^=。下面两条语句产生的最终作用相同:
val ^= 0377;
val = val ^ 0377;