逗号运算符扩展了for循环的灵活性,以便在循环头中包含更多的表达式。例如,下程序清单演示了一个打印一类邮件资费
/************************************************************************
功能:一类邮资
************************************************************************/
#include
int main(void)
{
const int FIRST_OZ = 46;
const int NEXT_OZ = 20;
int ounces,cost;
printf("价格成本\n");
for(ounces = 1, cost = FIRST_OZ;ounces <= 16;ounces++, cost += NEXT_OZ)
printf("%5d %4.2f(元)\n", ounces, cost / 100.0);
system("pause"); //暂停控制台
return 0;
}
该程序输出如下:
价格成本
1 0.46(元)
2 0.66(元)
3 0.86(元)
4 1.06(元)
5 1.26(元)
6 1.46(元)
7 1.66(元)
8 1.86(元)
9 2.06(元)
10 2.26(元)
11 2.46(元)
12 2.66(元)
13 2.86(元)
14 3.06(元)
15 3.26(元)
16 3.46(元)
请按任意键继续. . .
该程序在初始化表达式和更新表达式中使用了逗号运算符。初始化表达式中的逗号使ounces和cost都进行了初始化,更新表达式中的逗号使每次迭代ounces递增1、cost递增20(NEXT_Z的值是20)。绝大多数计算都在for循环头中进行(见下图)。
逗号运算符并不局限于在for循环中使用,但是这是它最常用的地方。逗号运算符有两个其他性质。首先,它保证了被它分隔的表达式从左往右求值(换言之,逗号是一个序列点,所以逗号左侧项的所有副作用都在程序执行逗号右侧项之前发生)。因此,ounces在cost之前被初始化。在该例中,顺序并不重要,但是如果cost的表达式中包含了ounces时,顺序就很重要。例如,假设有下面的表达式:
ounces++, cost = ounces * FIRST_OZ
在该表达式中,先递增ounce,然后在第2个子表达式中使用ounce的新值。作为序列点的逗号保证了左侧子表达式的副作用在对右侧子表达式求值之前发生。
其次,整个逗号表达式的值是右侧项的值。例如,下面语句x = (y = 3, (z = ++y + 2) + 5);的效果是:先把3赋给y,递增y为4,然后把4加2之和(6)赋给z,接着加上5,最后把结果11赋给 x。至于为什么有人编写这样的代码,在此不做评价。另一方面,假设在写数字时不小心输入了逗号:
这不是语法错误,C 编译器会将其解释为一个逗号表达式,即houseprice = 249 是逗号左侧的子表达式,500 是右侧的子表达式。因此,整个逗号表达式的值是逗号右侧表达式的值,而且左侧的赋值表达式把249赋给变量houseprice。因此,这与下面代码的效果相同:
houseprice = 249;
500;
记住,任何表达式后面加上一个分号就成了表达式语句。所以,500;也是一条语句,但是什么也不做。
另外,下面的语句
赋给houseprice的值是逗号右侧子表达式的值,即500。
逗号也可用作分隔符。在下面语句中的逗号都是分隔符,不是逗号运算符:
char ch, date;
printf("%d %d\n", chimps, chumps);
小结:新的运算符
赋值运算符:
下面的运算符用右侧的值,根据指定的操作更新左侧的变量:
+= 把右侧的值加到左侧的变量上
-= 从左侧的变量中减去右侧的值
*= 把左侧的变量乘以右侧的值
/= 把左侧的变量除以右侧的值
%= 左侧变量除以右侧值得到的余数
示例:
rabbits *= 1.6;与rabbits = rabbits * 1.6;相同
这些组合赋值运算符与普通赋值运算符的优先级相同,都比算术运算符的优先级低。因此,
contents *= old_rate + 1.2;
最终的效果与下面的语句相同:
contents = contents * (old_rate + 1.2);
逗号运算符:
逗号运算符把两个表达式连接成一个表达式,并保证最左边的表达式最先求值。逗号运算符通常在for循环头的表达式中用于包含更多的信息。整个逗号表达式的值是逗号右侧表达式的值。
示例:
for (step = 2, fargo = 0; fargo < 1000; step *= 2)
fargo += step;