C语言预处理_05
凡是以 “#”开头的均为预处理命令!
其定义的一般形式为:
#define 标示符 字符串
对于宏定义说明以下几点:
1.宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的代换,字符串中可以含任何字符,可以是常数,也可以是表达式,预处理程序对它不作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现!
2.宏定义不是说明或语句,在行末不需要加上分号,若加上分号则连分号一起置换。
3.宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如要终止其作用域可以使用 “#undef”命令。
4.宏名在源程序中若用引号括起来,则预处理程序不对其作宏代换。
5.宏定义允许嵌套,在宏定义的字符串中可以使用定义的宏名。在宏展开时由预处理程序层层代换
6.习惯上宏名用大写字母表示,以便于与变量区别!
7.可用宏定义表示数据类型,是书写方便。 例如 #define INTEGER int
#define 与 typedef 的区别:宏定义只是简单的字符串代换,是在预处理完成的,而typedef是在编译时处理的,它不是作简单的代换,而是对类型说明符重新命名。被命名的标示符具有类型定义说明的功能。typedef 后面会有分号,表示语句的结束。
一、无参数的宏定义
下面一段小程序,说明 #define 与 typedef 的区别:
#define PIN1 char* // 宏定义
typedef char* PIN2; // 重类型说明符重新命名 int main(void) {
PIN1 x, y;
PIN2 a, b;
printf("By #define : %lu %lu\n\n", sizeof(x), sizeof(y));
printf("By typedef : %lu %lu\n\n", sizeof(a), sizeof(b));
// 替换之后
/*
char *x, y; // x为一个指针, y是一个字符型
char *x, *y; // x, y均为指针类型
*/
return ;
}
打印结果:
By #define : 8 1 // 在我当前环境,指针类型占8个字节,char字符站1个字节 By typedef : 8 // x, y均为指针类型
编程中,适当使用宏定义可以提升我们的开发效率,但是慎用!
不用循环和递归,实现打印0-999:
#define BD(x) x;x;x;x;x;x;x;x;x;x;
int main(void) {
int n = ;
BD(BD(BD(printf("%d\n", n++))));
}
二、带参数的宏定义
C语言允许宏有参数。在宏定义中的参数称为形式参数,在宏调用中的参数称为实际参数。
对带参数的宏,在调用中,不仅要宏展开,而且要用实参去代换形参。
带参宏定义的一般形式: #define 宏名(形参表) 字符串
对于带参数的宏定义有以下问题需要注意:
1.带参数的宏定义,宏名和参数表之间不能有空格出现
例如:
#define MAX(a,b) ((a > b) ? a : b) // 正确写法
#define MAX (a,b) ((a > b) ? a : b) // 错误写法
2.在带参数宏定义中,形式参数不分配内存单元,因为不必作类型定义。而宏调用中的实参有具体的值。要用他们去代换形参,因此必须作类型说明符。
3.在宏定义中的形参是标示符,而宏调用中的实参可以使表达式。
4.在宏定义中,字符串中的形参通常要用括号括起来以避免出错。
#define PT(x) (x) * (x)
#define BT(x) x * x
int main(void) {
printf("PT=%d\nBT=%d\n", PT( + ), BT( + ));
}
5.带参数的宏和带参数的函数相似。但本质上不同!!
条件编译
预处理程序提供了条件编译的功能。可以按不同的条件去编译不同的程序部分,因而产生不同的目标代码文件。这对于程序的移植和调试时很有用的。不建议是用 /**/ ,因为C语言中不可以多行注释嵌套多行注释。建议使用预处理条件编译注释。
示例:
int main(void) {
#if 0
printf("-----------"); // 此处肯定不会执行
#endif
printf("Learning and record\n");
}
条件编译有三种潜规则:
第一种形式:
#ifdef 标示符
程序段1
#else
程序段2
#endif
或:
#ifdef 标示符
程序段2
#endif
功能:如果标示符已被 #define 命令定义过则对"程序段1"进行编译;否则对"程序段2"进行编译
第二种形式:
#ifndef 标示符
程序段1
#else
程序段2
#endif
功能:如果标示符没有被 #define 命令定义过则执行"程序段1";否则执行"程序段2"
场景:如果没有定义“标示符”,我们可以在“程序段1”位置定义“标示符”,以确保这个标示符的存在。
示例:
int main(void) {
// 若该宏未定义
#ifndef STRING
// 则给予定义
#define STRING "Learning and record……\n"
#endif
// 方便我们使用
printf("---%s", STRING);
}
第三种形式:
#if 常量表达式
程序段1
#else
程序段2
#endif
功能:如果“常量表达式”为真,则执行“程序段1”,反之执行“程序段2”
尊重作者劳动成果,转载请注明: 【kingdev】
最新文章
- Git : SSH 协议服务器
- 1-1 gulp 实战
- 关于CDN下查找网站真实ip
- Python数字,字符串
- css3 打勾 打叉
- UTF-8 BOM头
- maven配置httpclient3.X jar包
- WAMP(Windows、Apache、MySQL、php)安装配置过程常见问题
- sqlserver row_number 类似 mysql中 limit 用法
- Struts2 从一个Action跳至另一个Action
- 3d中的向量
- Codeforces Round #249 (Div. 2) C题,模拟画图 ----未解决!
- UVa 1210 (高效算法设计) Sum of Consecutive Prime Numbers
- dialog获取焦点
- JAVA用JNI方法调用C代码实现HelloWorld
- SwiftyUserDefaults-封装系统本地化的框架
- React-native 初始化项目很慢
- javascript项目实战之原生js模拟淘宝购物车
- 注册Docker官网账号 注册按钮不能点
- 定义类型别名(typedef,using)