C语言预处理

编译(编译器)

预编译/预处理(文本操作)

生成*.i的文件
1,#include
对于头文件的包含
2,注释的删除
使用空格替换注释
3,#define
对于定义的替换

预定义符号

__FILE__ 代码所在文件的绝对地址
__LINE__ 代码所在文件的行数
__DATE__ 代码执行的日期
__TIME__ 代码执行的时间
__FUNCTION__ 代码所在的函数名
__STDC__ 如果编译器遵循ANSI C,其值为1,否则未定义

#define

#开头都是预处理指令

#define尽量不要加;

#define 定义宏
#define机制包括一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义宏(define macro)
宏的声明方式

#define name(parament-list) stuff
//name与左括号必须紧紧相连

例子

#define SQUARE(X) X*X
int main(){
	int ret = SQUARE(5);
	//等价于int ret = 5*5;
	return 0;
}
#define SQUARE(X) X*X
int main(){
	int ret = SQUARE(5+1);
	//等价于int ret = 5+1*5+1;
	return 0;
#define SQUARE(X) (X)*(X)
int main(){
	int ret = SQUARE(5+1);
	//等价于int ret = (5+1)*(5+1);
	return 0;
#和##
#作用

将参数名传输到字符串中

#define PRINT(X) printf("the value of "#X" is %d",X)
int main(){
	int a = 10PRINT(a);
	//输出the value of a is 10
	return 0;
##作用

##可以把位于它两边的符号合并成一个符号,它允许宏定义从分离的文本片段创建标识符

#define CAT(A,B) A##B
int main(){
	int a8 = 8;
	printf("%d"CAT(a,8));
	//CAT(a,8)等价于字符串a8
}
带副作用的宏

牢记:宏的参数是直接替换进去而不是算好了之后再带进去的

#define MAX(X,Y) ((X)>(Y)?(X):(Y))
int main(){
	int a = 10;
	int b = 20;
	int max = MAX(a++,b++)//12
	printf("%d",a);//11
	printf("%d",b);//13
}
属性#define函数
代码长度每次使用时,宏代码都会被插入到程序中。除了非常小的宏之外,程序的长度会大幅度增长函数代码只出现于一个地方﹔每次使用这个函数时,都调用那个地方的同一份代码
执行速度更快存在函数的调用和返回的额外开销,所以相对慢一些
操作符优先级宏参数的求值是在所有周围表达式的上下文环境里,除非加上括号,否则邻近操作符的优先级可能会产生不可预料的后果,所以建议宏在书写的时候多些括号。函数参数只在函数调用的时候求值一次,它的结果值传递给函数。表达式的求值结果更容易预测。
带有副作用的参数参数可能被替换到宏体中的多个位置,所以带有副作用的参数求值可能会产生不可预料的结果。函数参数只在传参的时候求值一次,结果更容易控制。
参数类型宏的参数与类型无关,只要对参数的操作是合法的,它就可以使用于任何参数类型。函数的参数是与类型有关的,如果参数的类型不同,就需要不同的函数,即使他们执行的任务是不同的。
调试宏是不方便调试的函数是可以逐语句调试的
递归宏是不能递归的函数是可以递归的
命名约定

宏名全部大写,函数名不要全部大写

#undef

这条指令用于移除一个宏定义

#define MAX 100
int main(){
printf(""%d,MAX);
#undef MAX
printf(""%d,MAX);//什么都不会输出出来
}
条件编译
#DEBUG 1;
int main(){
	int a = 10;
#ifdef DEBUG//当宏没有定义时,下面的语句不会进行编译
	printf("%d",a);
#endif		
}

#DEBUG 1;
int main(){
	int a = 10;
#ifndef DEBUG//当宏没有定义时,下面的语句进行编译
	printf("%d",a);
#endif		
}

#if 常量表达式
//语句
#endif
//常量表达式由预处理器求值,为真则编译语句,反之则不编译

多个分支的条件编译

#if 常量表达式
//语句
#elif 常量表达式
//语句
#else 
//语句
#endif

判断是否被定义

#if define(symbol)//定义过symbol,编译下面语句
//语句
#ifde symbol

#if !define(symbol)//没定义过symbol,编译下面语句
//语句
#ifdef symbol
文件包含

头文件被包含方式
#include "filename"
查找策略:先在源文件所在目录下查找,如果头文件未找到,编译器就像查找库函数头文件一样在标准位置查找头文件,如果找不到就显示编译报错

#include <filename>
查找策略:直接去标准路径下去查找,如果找不到就提示编译报错

解决头文件重复包含问题

#ifndef _TEST_H_
#define  _TEST_H_
//头文件内容
#endif 

或者

#pragma once

编译

把C语言代码翻译成汇编代码
编译成*.s文件

1,语法分析
2,词法分析
3,语义分析
4,符号汇总(函数名,全局变量)

这个阶段与《编译原理》有关

汇编

将汇编指令转化为二进制指令
形成符号表
生成*.o文件或*.obj文件

链接(链接器)

1,合并段表
2,符号表的合并和符号表的重定位

过程操作

1,预处理选项

gcc  -E test.c -o test.i 

预处理之后产生的结果都放在test.i文件中

2,编译选项

gcc -s test.c

编译完成之后就停下路,结果保存在test.c
3,汇编

gcc -c test.c

汇编王朝之后停下来,结果保存在test.o

运行环境

1.程序必须载入内存中。在有操作系统的环境中︰一般这个由操作系统完成。在独立的环境中,程序的载入必须由手工安排,也可能是通过可执行代码置入只读内存来完成。
2,程序的执行便开始。接着便调用main函数
3.开始执行程序代码。这个时候程序将使用一个运行时堆栈( stack ),存储函数的局部变量和返回地址。程序同时也可以使用静态 ( static )内存,存储于静态内存中的变量在程序的整个执行过程一直保留他们的值。
4.终止程序。正常终止main函数;也有可能是意外终止。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值