参考: 里科《C和指针》
语句
C没有bool,是用整型代替的。因此if(expression)中,expression只要是可以产生整型结果即可,且0表示假,非0表示真。
如果有只靠缩进没有{}的else语句,会自动匹配离它最近的、不完整的if语句。
while
int num = 0;
/*
** Get the numbers, stopping at eof or when a number is < 0.
*/
while( num < max && scanf( "%d", &columns[num] ) == 1
&& columns[num] >= 0 )
num += 1;
// 注意:scanf函数的标量参数前需要添加&
do while
do
statement
while( expression );
单独一个分号是空语句。
goto尽可能不用。用的话需要保证标签(标识符+冒号)唯一
while( exp1 ) {
while( exp2 ) {
while( exp3 ) {
if ( exp4 )
goto quit;
}
}
}
quit: ;
/ 只有两个操作数都是整数时才执行整除,否则是浮点数除法
有符号数可能存在算术右移的情况,即空缺位由符号位填充(以保持符号不变),到底是采用算术右移还是逻辑右移(空缺位填0)是由编译器决定的,因此该操作的引入可能导致代码不可移植。
+=和+的区别在于,前者的左操作数只求值一次,因此应该尽量使用复合赋值符
++a = 10;为什么是错的?
因为++的结果是a的拷贝,是值,所以不能给值赋值。
编译器只要不违背优先级和结合性原则,就可以自由决定复杂表达式的求值顺序,如果表达式的结果依赖于求值的顺序,那么它本质上就是不可移植的。
因为优先级决定的是相邻操作符的执行顺序,结合性是左右顺序。比如 a * b + c * d + e * f 中,虽然乘法比加法优先,但是是先计算 a * b 还是先计算 c * d并没有规定。
-
位运算
1)将指定的位置设置为1
value |= 1 << bit_number;
2)将指定位置清0
value &= ~ ( 1 << bit_number );
3)计算参数值中1的位的个数
int count_one_bits( unsigned value ) { int ones; for (ones = 0; value != 0; value >>= 1) { if (value & 1) ones++; } return ones; }
条件操作符
if ( a > 5 )
b[ 2 * c + d( e / 5) ] = 3;
else
b[ 2 * c + d( e / 5) ] = -20;
// 改成
b[ 2 * c + d( e / 5) ] = a > 5 ? 3 : -20;
逗号操作符:从左向右逐个求值,整个逗号表达式的值就是最后那个表达式的值。使用场景如下,当a需要经过前两个语句计算出结果,再进行判断时,就可以使用逗号操作符。不过如果不能提升编程效率或者无法保证正确,则不要使用。
while( a = get_value(), count_value( a ), a > 0 ) {}
也可以写成
while( count_value( get_value() ), a > 0 ) {}
隐式类型转换
C的整型算术运算总是以default类型的精度来进行的,所以下面的char(还有short)在使用之前会被转换为int,这个称为整型提升(integral Promotion),b和c会变成int,加完得到结果,截短再存到a中。
char a, b, c;
...
a = b + c;