一、程序的翻译环境和执行环境
翻译环境:在这个环境中,源代码被转换为可执行的机器指令(可执行文件)。
执行环境:实际执行代码的环境(可执行文件中的内容执行的环境)。
在翻译环境中,组成一个程序的每个源文件通过编译过程分别转换成对应的目标文件,然后每个目标文件由链接器捆绑在一起,形成一个单一且完整的可执行程序。
编译器编译分为三个阶段:
二、预处理
2.1、预定义符号
例子:
printf("file:%s line:%d\n", __FILE__, __LINE__);
执行结果:
2.2、#define
2.2.1、#define定义的宏
用于对数值表达式进行求值的宏在定义时应多加括号,避免在使用宏时因操作符的相互作用而出现不可预料的结果。
多加括号:
#define DOUBLE(x) ((x) + (x))
2.2.2、#
int main()
{
printf("hello ""world\n");
return 0;
}
执行结果:
从以上例子看出字符串有自动连接的特点。
这时当字符串作为宏参数的时候,可以把字符串放到字符串中:
#define PRINT(FORMAT, VALUE) \
printf("the value is "FORMAT"\n", VALUE)
int main()
{
PRINT("%d", 10);
return 0;
}
参数替换之后,printf(“the value is “FORMAT”\n”, VALUE)变成printf(“the value is “”%d”“\n”, 10),即printf(“the value is %d\n”, 10)。
另外一个方法是使用#,把一个宏参数转换成对应的字符串:
#define PRINT(FORMAT, VALUE) \
printf("the value of "#VALUE" is "FORMAT"\n", VALUE)
int main() //printf("the value of "#VALUE" is "FORMAT"\n", VALUE)
{ //变成printf("the value of ""VALUE"" is "FORMAT"\n", VALUE),
int i = 10; //然后变成printf("the value of ""i + 3"" is ""%d""\n", i + 3),
PRINT("%d", i + 3); //即printf("the value of i + 3 is %d\n", i + 3)。
return 0;
}
代码中的#VALUE会预处理为"VALUE"(#i + 3预处理为"i + 3"),printf(“the value of “#VALUE” is “FORMAT”\n”, VALUE)变成printf(“the value of ““VALUE”” is “FORMAT”\n”, VALUE),然后变成printf(“the value of ““i + 3"” is “”%d””\n", i + 3),即printf(“the value of i + 3 is %d\n”, i + 3)。
2.2.3、##
##可以把位于它两边的符号合成一个符号,这样的连接必须产生一个合法的标识符,否则其结果是未定义的。
#define ADD_TO_SUM(num, value) \
sum##num += value
int main()
{
int sum5 = 5;
ADD_TO_SUM(5, 10);//作用是:给sum5增加10
return 0;
}
2.2.4、带副作用的宏参数
当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么在使用这个宏的时候就有可能出现不可预测的结果。副作用就是表达式求值时出现的永久性效果。
x + 1;//不带副作用
x++;//带副作用
有问题的代码:
#define MAX(a, b) ((a) > (b) ? (a) : (b))
int main()
{
int x = 5;
int y = 8;
int z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);
return 0;
}
有问题的代码的执行结果: