预处理
在程序预处理阶段进行执行
利用宏创建字符串、宏拼接、可变参数宏
创建字符串:#
#define X(x) #x printf(X(dawdwadad));//dawdwadad--->"dawdwadad"
拼接字符串:##
#define X(x) x##"adwada" X(awdad);//-->awdadadwada
可变参数宏:…
#define PRINT(...) printf(__VA_ARGS__)//...表示参数可变 //__VA_ARGS__ 用来获取可变参数 PRINT("awdaw%d",788);
取消宏定义、#if #elif #else #endif
取消宏定义:#undef
#define X123 #undef X//取消后面就不能用了
条件编译:#if #elif #else #endif
#if 0//假,不执行 int b = 0; #elif 1//真,执行 int a = 0; #else//其他 int c = 0; #endif b = 2; //未定义错误 a = 2; //正常 c = 2; //未定义错误
全文注释
#if 0//可用于全文注释
#ifdef #endif
#ifdef _H_//如果定义了_H_ //各种声明定义 int a; #endif//结尾 a;//未声明标识符
TIME、DATA、FILE、LINE
printf(__TIME__);//当前时间 printf(__DATA__);//当前日期 printf(__FILE__);//当前文件完整路径 printf(__LINE__);//宏所在行数
STDC_VERSION、STDC、STDC_HOSTED
__STDC_VERSION__ //当前 C 语言标准版本,工程属性->C 语言版本 老旧版本不支持 199409L(C95) 201112L(C11) 201112L(C11) 201710L(C17) __STDC__ //如果当前有它并指定为 1,说明是符合 C 标准,但是没有这个东西 __STDC_HOSTED__ //如果当前环境有它并为 1,说明支持完整的标准库
#Line、#error
#line 5 "text.c"//如果:这行为0,但是你这行改了,下一行就变5了,依次类推:0-->5--->6.... #error not hello //如果有错误下划线等,这是正常的 //后面的是保存信息
pragma、宏拼接
pragma
#pragma once//防止头文件重复包含 #pragma pack(4)//设置字节对齐 #pragma message("hello")//编译时显示字符串 #pragma comment(lib,库名)//包含库 #pragma warning(once:4996)//#pragma warning(once: 4996 ) //#pragma warning(error: 4996) 禁用警告,错误4996 //设置其下变量的使用权限 #pragma section("mycode",read)//read只读 //read,write可读可写 __declspes(allocate("mycode"))//字符串填写上面指定的 int i = 0;//i只读 #define X 1//以栈结构存储宏名 #pragma push_macro("X")//入栈 #pragma pop_macro("X")//出栈 #pragma region//设置编译器内代码折叠 code//该部分可折叠 #pragma endregion //共享数据段 #pragma data_seg("data") int a = 12; #pragma endregion
宏拼接
一个宏名仅是其同一行的名字,跟别的行没有关系了
#define PRINT int i; for (i = 0; i < 5; i++) { printf("%d ", i); }
我想让 PRINT 代替整个 5 行代码,其做不到,其本体就是 int i;,因为宏只把同一行的代码 作为本体,所以可以写到一行,如下:
#define PRINT int i;for (i = 0; i < 5; i++){ printf("%d ", i); }
但是,这样写法代码完全看不出结构,代码一多,那就瞎了,这就有了宏拼接符:“”
#define PRINT int i; \ for (i = 0; i < 5; i++) \ { \ printf("%d ", i); \ }
他的意义就是告诉编译器上一行与下一行是链接的,紧挨着的.