C语言#define关键字,条件编译及预定义符号知识详解
程序员文章站
2022-09-28 14:12:01
一.预定义符号
C标准定义了如下五个预处理符号
__FILE__//进行编译的源文件名称
__LINE__//文件当前行号
__DATE__//文件被编译的日期
__...
一.预定义符号
C标准定义了如下五个预处理符号
__FILE__//进行编译的源文件名称 __LINE__//文件当前行号 __DATE__//文件被编译的日期 __TIME__//文件被编译的时间 __STDC__//如果编译器遵循ANSI C,值为1,否则未定义
在文件输入输出以及输出日志等场合下我们会用到这些预处理符号,比如
#include #include #define LOG fprintf(pf,"%d %d %d %d\n",\ __FILE__,__LINE__,__DATE__,__TIME__,__STDC__) int main() { int i = 0; FILE* pf = fopen("log.txt","w");w if(pf == NULL) { printf("open failure!\n"); exit(EXIT_FAILURE); } for(i = 0; i < 5; i++) { LOG; } fclose(pf); return 0; }
\+回车 : 续行符,转义字符,后面不能加任何除回车之外的字符,否则会报错
二.#define关键字
#define有两个作用,一个是定义标识符,第二个是定义宏
1.定义标识符
//#define name stuff #define MYINT int //typedef stuff name typedef int MYINT;
不要在define定义的标识符里加分号
2.定义宏
#define机制包括一个规定,允许把参数替换到文本中,这种实现通常叫做宏或定义宏
//#define name(parament-list) stuff #define MAX ((a)>(b)?(a)(b)) int a = MAX(11+2,11-2);//int a = ((11+2)>(11-2)?(11+2):(11-2));
宏定义永远不要吝啬括号,防止因为优先级的问题导致出现不可思议的后果 宏不能实现递归 宏只文本替换,并不简化计算 字符串中的内容不会发生宏替换
#define X 100 printf("X");//不会宏替换#和##
3.#和 ##
编译器会自动就两个相邻字符串自动连接成一个
#:把参数插入到字符串中
#a -> “a”
#define PRINT(x,format)\ printf("the value of "#a" is "#format"\n",x) //printf("the value of ""x"" is""format""\n",x) int a = 10; PRINT(x,"%d");
##可以把位于它两边的符号拼成一个符号
#define CAT(x,y) ((x)##(y)) int num5 = 10; CAT(num,5);//=num5
4.宏和函数对比
①宏没有类型检查
没有类型检查是宏的一大特点(但同时也是不安全的隐患),可以完成很多函数不可能做到的功能
#define MALLOC(n,type)\ (type*)malloc(n*sizeof(type))
②宏没有函数的调用和返回等额外开销,效率更高,
但是当逻辑复杂并且代码量巨大时,调用和返回的开销和运算的时间相比微乎其微,可以省略
③宏单纯只是文本替换,不能进行调试,很致命
④宏可能会使代码长度增长
⑤函数的参数如果是一个表达式,会计算出表达式的结果传入参数
宏的参数如果是一个表达式,只会文本替换,不会进行计算
总结:各有千秋,小代码量使用宏,大代码量使用函数,C++中摈弃宏这一概念,使用内联,基于优缺点,最好不要使用宏
5.带有副作用的宏参数
int a = 10;int b = 0; b = a+1; b = a++;//带有副作用的表达式,使a发生改变
不要给宏参数传入带有副作用的参数
6.命名约定
宏:全部大写
函数: 不全部大写
例外
//getchar()由库函数实现还是由宏实现是编译器决定的 #define getchar() getc()
#undef name //取消宏标识符name的定义
三.条件编译
1.如果定义了DEBUG,语句执行,反之,不执行
#define __DEBUG__ 1 #ifdef __DEBUG__ 语句; #endif
2.条件编译
#if 常量表达式(为真编译,为假,不编译) 语句; #endif
3.多分支条件编译
#if 常量表达式 语句; #elif 常量表达式 语句; #elif 常量表达式 语句; #endif
3.判断是否被定义
#ifndef __DEBUG__ //==>#if !defined(__DEBUG__) #ifdef __DEBUG__ //==>#if defined(__DEBUG__)
4.嵌套指令
预处理指令可以进行嵌套
pragma onec //新式防止头文件被多次引用 //繁琐 #ifndef __TEST_H__ #define __TEST_H__ #endif //__TEST_H__
四.其它
1.头文件的包含
#include //直接去库目录下查找 #include "filename" //先查找当前工作目录下查找,然后去库目录查找
2.#error “…”
遇到#error便会产生一个编译错误
3.strcpy();
源字符串必须以’/0’结束,返回目标字符串地址
且这个函数不管目标空间是否放得下返回的目标字符串
4.assert(条件语句)
善用断言,头文件为
5.链式反应
这次函数的返回值可以作为下一个函数的参数,达到链式反应的效果
printf(“%d\n”,strlen(strcpy(p,”hello world!”)));
6.常见错误:
编译型错误
直接看编译器错误提示信息即可 链接型错误
函数名,变量名写错,库文件使用错误(第三方库) 运行时错误