17_++和–操作符分析
1、++和–分析
++和–操作符分别对应着两条汇编指令;
前置:变量自增(减)1,取变量值;
后置:取变量值,变量自增(减)1;
程序示例:
#include <stdio.h>
int main()
{
int i = 0;
int r = 0;
r = (i++) + (i++) + (i++);
printf("i = %d\n", i);
printf("r = %d\n", r);
r = (++i) + (++i) + (++i);
printf("i = %d\n", i);
printf("r = %d\n", r);
return 0;
}
输出结果:
VC编译器中:i = 3, r = 0; i = 6, r =18;
GCC编译器中:i = 3, r = 3; i = 6, r = 16;
结果分析:
在C语言中只规定了++和–对应指令的相对执行次序;++和–对应的汇编指令不一定连续
运行;在混合运算中,++和–的汇编指令可能被打断执行;
2、面试题
#include <stdio.h>
int main()
{
int i = 0;
int j = ++i+++i+++i; //error
int a = 1;
int b = 4;
int c = a+++b; //a++ + b
int* p = &a;
b = b/*p; // /*变成注释
printf("i = %d\n", i);
printf("j = %d\n", j);
printf("a = %d\n", a);
printf("b = %d\n", b);
printf("c = %d\n", c);
return 0;
}
解决办法:加上空格即可;
空格可以作为C语言中一个完整符号的休止符,编译器读入空格后
立即对之前读入的符号进行处理。
3、小结
(1)++和–操作符在混合运算中的行为可能不同;
(2)编译器通过贪心法处理表达式中的子表达式;
(3)空格可以作为C语言中的一个完整符号的休止符;
(4)编译器读入空格后立即对之前读入的符号进行处理;
18_三目运算符和逗号表达式
1、三目运算符
a?b:c
若a为真,返回b的值;否则,返回c的值。
**注意:**三目运算符返回的是值,而不是变量本身;
错误程序举例
(a<b?a:b) = 3; //error
那么三目运算符的返回类型是什么?
(1)通过隐式类型转换规则返回b和c中的较高类型;
(2)当b和c不能隐式转换到同一类型时将编译出错;
2、逗号表达式
逗号表达式主要用于将多个子表达式连接为一个表达式,其值为最后一个子表达式的值,
前N-1个子表达式可以没有返回值,逗号表达式按照从左向右的顺序计算每个子表达式的值。
程序示例:
#include <stdio.h>
void hello()
{
printf("Hello!\n");
}
int main()
{
int a[3][3] = {
(0, 1, 2), //相当于2
(3, 4, 5), //相当于5
(6, 7, 8) //相当于8
};
int i = 0;
int j = 0;
while( i < 5 )
printf("i = %d\n", i),
hello(),
i++; //相当于printf("i = %d\n", i),hello(),i++;
for(i=0; i<3; i++)
{
for(j=0; j<3; j++)
{
printf("a[%d][%d] = %d\n", i, j, a[i][j]);
}
}
return 0;
}
输出结果为:
3、面试题:一行代码实现strlen
#include <stdio.h>
#include <assert.h>
int strlen(const char* s)
{
return assert(s), (*s ? strlen(s + 1) + 1 : 0);
}
int main()
{
printf("len = %d\n", strlen("Delphi"));
printf("len = %d\n", strlen(NULL));
return 0;
}
4、小结
(1)三目运算符返回变量的值,而不是变量本身;
(2)三目运算符通过隐式类型转换规则确认返回值类型;
(3)逗号表达式按照从左向右的顺序计算每个子表达式的值;
(4)逗号表达式的值为最后一个子表达式的值;