当前位置: 首页 > news >正文

C语言宏定义详解

文章目录

  • 宏定义
    • 无参宏定义
    • 带参宏定义
    • 固定参数宏
    • 可变参数宏
  • 多语句宏处理
  • 连接符
  • 条件判断
  • 常见预定义宏

宏在C语言中是一段有名称的代码片段(使用#define定义),在预处理阶段会把程序中的宏名替换为对应的代码片段,然后才进入编译阶段由编译器进行编译。

  • #define:宏定义
  • #undef:取消宏定义
  • #ifdef:判断宏是否定义

宏定义

无参宏定义

最简单的宏定义时间用指定的标识符来代表代码片段,当宏定义有多行时,需要在行尾使用\来连接。

#define 宏名称  代码片段// 如(多行时,行尾使用\作为续行符)
#define BUFFER_SIZE 1024
#define RANDOM (2.0*(double)rand() / RAND_MAX)
#define NUMBERS 1, \2, \3int x[] = { NUMBERS };

带参宏定义

类似函数定义,宏名称后紧跟括号(名称与括号间不能有空格)。

固定参数宏

“形参列表”是用逗号隔开的多个标识符(也可以空,表示无参数),实参列表中的实参数量必须与宏定义中的形参数量一样多。

#define 宏名称( [形参列表] )  替换文本
// 如
#define GETCHAR() getc(stdin)
#define MAX(a,b) ((a)>(b)?(a):(b))
//参数要用括号括起,避免实参为表达式时出错,如
#define MULTIPLY(a, b) ((a)*(b))
//MULTIPLY(1+2, 3+4)会预处理为((1+2)*(3+4)),若不加括号,则会变为1+2*3+4(意义全变)。

即使加括号,宏在一定情况下还是会产生二义性(要避免此类使用),如

#define SQUARE(a) ((a)*(a))

SQUARE(a++)会预处理为((a++)(a++))(若a=3,则结果为34)。

可变参数宏

C99标准允许定义有省略号的宏(在替换文本中使用__VA_ARGS__标识可变参数),省略号必须放在参数列表的最后面表示可变参数。

#define OUTLOG(fmt, ...) printf(__func__, fmt, __VA_ARGS__)

多语句宏处理

对于有多条语句的复杂宏,为避免错误,需要使用do{}while(0)包含(避免出现多余分号等的报错)。

#define SAFE_DELETE(ptr)  \do{ \if(NULL != ptr){  \delete ptr;     \ptr=NULL;       \} \}while(0)

连接符

在宏体中,通过在参数前面加不同符号,会有不同的扩展:

  • #:参数扩展为字符串形式
  • ##:连接符,把两部分连接为一个标识符
  • #@:参数扩展为字符
    通过这些连接符,可以实现一些复杂的操作(如MFC中自动生成类等):
#define CHAR_VAR(n, c)  char var##n = #@c
#define ARRAY_VAR(n, str)  char ary##n[] = #str// char vara='a';
CHAR_VAR(a, a);
// char arya[]="abc";
ARRAY_VAR(a, abc);

条件判断

条件编译里面有判断语句,如 #if 、#else 、#elif 及 #endif。类似if语句,在里面可以使用与、或、非及比较表达式。通过条件判断可以使代码适应不同的运行环境(针对不同环境,编译不同代码部分)。

//#ifdef 是 '#if defined'的简写
#if defined(X_OS_WIN32) || defined(X_OS_WIN64) 
// Windows代码处理...
#endif

VC中判断当前是否为Debug:

#ifdef _DEBUG
#pragma comment(lib, "my-debug.lib")
#else
#pragma comment(lib, "my-release.lib")
#endif

C++中定义标准C接口(使用extern “C”)

#ifdef __cplusplus
extern "C"{
#endif// ... 代码#ifdef __cplusplus
}
#endif
在判断是否满足指定要求#if defined(MY_TEST_VER) && (MY_TEST_VER>=2)
// ...
#endif

常见预定义宏

预定义宏:

__DATE__: 字符串, 进行预处理的日期("Mmm dd yyyy", 如May 27 2006)
__TIME__: 字符串, 源文件的编译时间("hh:mm:ss",09:11:10)
__FILE__: 字符串, 代表当前源代码文件名(包含详细路径, 如F:/a.c)
__LINE__: 整数值, 代表当前源代码文件中的行号
__STDC__: 布尔值,1时表示该实现严格遵循ANSIC标准
__STDC_VERSION__: 长整型值, 表示编译器所遵循的C标准的版本号(yyyymmL,199101L)
__func__: 字符串, 当前所在函数名(C99标准)
__PRETTY_FUNCTION__: 在C中,__func__; 而在C++, 则记录了当前函数的头信息
__VA_ARGS__: 保存了可变参数列表 "…"
__cplusplus: 长整型值, 表示了C++的版本号(yyyymmL,199711L)
http://www.lryc.cn/news/127645.html

相关文章:

  • SwiftUI 动画进阶:实现行星绕圆周轨道运动
  • 物理试题-空气净化器
  • Es、kibana安装教程-ES(二)
  • leetcode 917.仅仅反转字母
  • 有没有推荐的golang的练手项目?
  • springBoot的日志文件
  • Linux学习之iptables的nat表
  • 【数据结构】 ArrayList简介与实战
  • 您的网站不应该只提供一套通用 API
  • vue tree禁用和多选变为单选
  • ES6新特性。对象、数组新增方法
  • request发送http请求
  • leaflet实现MARK指向的方向随机
  • 如何使用Python编写小游戏?
  • 【Leetcode】84.柱状图中最大的矩形(Hard)
  • Arraylist集合
  • https的原理和方案
  • VTK 判断一个 点 是否在一个模型 stl 内部 vtk 点是否在内部 表面 寻找最近点
  • 【数据结构OJ题】链表的回文结构
  • Nginx常见的三个漏洞
  • 爬虫逆向实战(十六)--某建筑市场平台
  • 用Python做一个滑雪小游戏
  • EXCEL按列查找,最终返回该列所需查询序列所对应的值,VLOOKUP函数
  • java编译报错,get方法报错
  • 可以降低CPU负载的网络传输技术——LSO
  • [管理与领导-25]:IT基层管理者 - 团队管理 - 如何留人, 如何留住关键人才
  • 【Redis】Redis 的学习教程(二)之 Jedis
  • VB+SQL银行设备管理系统设计与实现
  • Python系统学习1-9-类一之类语法
  • PHP“深入浅出”淘宝商品详情数据接口获取方法,淘宝API申请指南