2.6.2 复合赋值表达式
在赋值运算符之前加上其他运算符可以构成复合赋值运算符。C语言规定可以使用10种复合赋值运算符,其中与算术运算有关的复合赋值运算符有:+=、-=、*=、/=、%=(注意:两个符号之间不可以有空格)。复合赋值运算符的优先级与赋值运算符的优先级相同。表达式n+=1的运算规则等价于n=n+1,表达式n*=m+3的运算规则等价于n=n*(m+3),因为运算符“+”的优先级高于复合赋值运算符“*=”。其他以此类推。
例2.3 已有变量a,其值为9,计算表达式a+=a-=a+a的值。
因为赋值运算符与复合赋值运算符“-=”和“+=”的优先级相同,且运算方向自右至左,所以:
(1)先计算“a+a”;因a的初值为9,所以该表达式的值为18,注意a的值求变。
(2)再计算“a-=18”,此式相当于“a=a-18”,因a的值仍为9,所以表达式的值为-9,注意a的值已为-9。
(3)最后计算“a+=-9”,此式相当于“a=a+(-9)”,因a的值此时已是-9,所以表达式的值为-18。
由此可知,表达式a+=a-=a+a的值是-18。
2.6.3 赋值运算中的类型转换
在赋值运算中,只有在赋值号右侧表达式的类型与左侧变觅类型完全一致时,赋值操作才能进行。如果赋值运算符两侧的数据类型不一致,在赋值前,系统将自动先把右侧表达式求得的数值按赋值号左边变量的类型进行转换,也可以用强制类型转换的方式人为地进行转换后将值赋给赋值号左边的变量。这种转换仅限于数值数据之间,通常称为“赋值兼容”。对于另外一些数据,例如后面将要讨论的地址值就不能斌给一般的变量,称为“赋值不兼容”。
在这里,特别需要指出的是在进行混合运算时整型数据类型之间的转换问题。
在C语言的表达式(不包括赋值表达式)中,如果运算符两边的整数类型不相同,将进行类型之间的转换。转换规则如下:
(1)若运算符两边一个是短整型,一个是长整型,则将短整型转换为长整型,然后进行运算。
(2)若运算符两边一个是有符号整型,一个是无符号整型;则将有符号整型转换成无符号整形,然后进行运算。
在C语言的赋值表达式中,赋值号右边的值先转换成与赋值号左边的变量相同的类型,然后进行赋值。应当注意:
(1)当赋值号左边的变量为短整型,右边的值为长整型时,短整型变量只能接受长整型数低位上两个字节中的数据,高位上两个字节中的数据将丢失。也就是说,右边的值不能超出短整型的数值范围,否则将得不到预期的结果。例如,若有以下定义和语句:
short a;
unsigned long b;
b=98304; a=b;
printf("%d\n",a);
则a中的值为-32768。因为98304(二进制数11000000000000000)已经超出短整型的数值范围(-32768~32767),a截取b中低16位中的值(二进制数1000000000000000),由于最高位为1,因此。中的值为-32768。
(2)当赋值号左边的变量为无符号整型,右边的值为有符号整型时,则把内存中的内容原样复制。右边数值的范围不应超出左边变量可以接受的数值范围。同时器要注意,这时负数将转换为正数。例如,变量。被说明为unsigned类型,在进行了a=-1的赋值操作后,将使a中的值为65535。
(3)当赋值号左边的变量为有符号整型,右边一的值为无符号整型时,复制的机制同上。这时若符号位为1,将按负数处理。