C语言宏函数(Macro Functions)通过预处理实现代码复用,凭借零函数调用开销、类型无关性等特性,成为嵌入式开发、性能敏感场景的必备工具。以下精选10个经典宏函数,覆盖类型检查、安全操作、调试打印等核心场景,均遵循"高效简洁、安全可控"原则。
// 避免表达式副作用(如MAX(a++, b++))的安全实现
//GNU
#defineMAX(a, b)({\__typeof__(a) _a =(a);\__typeof__(b) _b =(b);\_a > _b ? _a : _b;\})//GNU#defineMIN(a, b)({\__typeof__(a) _a =(a);\__typeof__(b) _b =(b);\_a < _b ? _a : _b;\})
优势:通过中间变量_a/_b缓存表达式结果,解决传统MAX(a,b)((a)>(b)?(a):(b))的重复计算问题(如MAX(i++, j++)会导致i/j自增两次)。
使用:int max_val = MAX(3+2, 10/2); // 结果5
//GNU#defineABS(x)({\__typeof__(x) _x =(x);\_x <0?-_x : _x;\})
特性:通过__typeof__自动适配int/float/double等类型,例如:
ABS(-5);// 5 (int)
ABS(-3.14);// 3.14 (double)
#defineARRAY_SIZE(arr)(sizeof(arr)/sizeof((arr)[0]))
用途:获取数组元素个数,避免硬编码长度(防止数组扩容后忘记更新):
int buf[10];
for(int i =0; i <ARRAY_SIZE(buf); i++){...}// 循环10次
陷阱:不可用于指针(如int *p = buf; ARRAY_SIZE(p)会计算sizeof(p)/sizeof(int),结果为1或2,导致错误)。
#defineMEM_ZERO(p, type)memset(p,0,sizeof(type))
优势:相比直接memset(p,0,sizeof(p))(若p为指针则错误),强制指定类型,确保清零大小正确:
typedefstruct{int a;char b;} Data;
Data d;
MEM_ZERO(&d, Data);// 正确清零整个结构体(sizeof(Data)=8字节)
// 设置bit n(置1)
#defineBIT_SET(reg, n)((reg)|=(1U<<(n)))
// 清除bit n(置0)
#defineBIT_CLR(reg, n)((reg)&=~(1U<<(n)))
// 翻转bit n
#defineBIT_TOGGLE(reg, n)((reg)^=(1U<<(n)))
// 检查bit n是否为1
#defineBIT_IS_SET(reg, n)(((reg)>>(n))&1U)
嵌入式场景:控制GPIO寄存器(以STM32为例):
BIT_SET(GPIOA->ODR,5);// PA5引脚置高(点亮LED)
BIT_CLR(GPIOB->IDR,3);// 清除GPIOB->IDR的bit3
if(BIT_IS_SET(GPIOC->IDR,10)){...}// 检查PC10引脚是否输入高电平
// 从reg中提取[start, end]位(包含start和end,0表示最低位)
#defineBITS_EXTRACT(reg, start, end)({\uint32_t _mask =((1U<<((end)-(start)+1))-1)<<(start);\((reg)& _mask)>>(start);\})
用途:解析寄存器中的状态字段(如ADC结果、传感器数据):
uint32_t adc_reg =0x12345678;// 假设ADC结果存于bit15~bit4
uint16_t adc_val =BITS_EXTRACT(adc_reg,4,15);// 提取bit4~bit15,结果0x3456
#include<stdio.h>
// 打印文件名、行号、函数名和自定义消息
#defineDEBUG_PRINT(fmt,...)\printf("[%s:%d %s()] "fmt "\n",__FILE__,__LINE__,__func__,##__VA_ARGS__)
效果:输出包含上下文的调试信息,便于定位问题:
DEBUG_PRINT("Initializing UART, baudrate=%d",115200);
// 输出:[main.c:42 main()] Initializing UART, baudrate=115200
扩展:可替换printf为日志函数(如log_info),适配嵌入式无stdio环境。
// NDEBUG未定义时启用断言(Release模式可通过-DNDEBUG禁用)
#ifndefNDEBUG
#defineASSERT(expr)do{\if(!(expr)){\printf("ASSERT FAILED: %s at %s:%d %s()\n", #expr,__FILE__,__LINE__,__func__);\while(1);// 死循环或触发硬件异常 \
} \
} while(0)
#else
#defineASSERT(expr)((void)0)// Release模式空操作
#endif
用途:验证关键假设(如指针非空、数组越界):
voidprocess_data(int*buf,int len){
ASSERT(buf !=NULL);// 确保指针有效
ASSERT(len >0);// 确保长度合法
// ...
}
注意:断言不可用于处理预期错误(如文件打开失败),仅用于捕获编程错误。
// 将循环展开为N次独立操作(适用于小N,如N=4)
#defineUNROLL_LOOP(N, func)do{\switch(N){\case4:func(3);/* fallthrough */\case3:func(2);/* fallthrough */\case2:func(1);/* fallthrough */\case1:func(0);break;\default:break;\}\}while(0)
性能优化:减少循环控制开销(如数组初始化):
int arr[4];
// 展开为4次赋值,避免i++和条件判断
UNROLL_LOOP(4,(void(*)(int))(void(*)(int))((void(*)(int))(void(*)(int))( \
[&](int i){ arr[i]= i *2;})
));// arr结果:[0,2,4,6]
// 包装函数并记录调用次数
#defineCOUNTED_FUNC(name, ret,...)\staticint name##_count =0;\ret name(__VA_ARGS__){\name##_count++;\/* 原函数逻辑 */\}\int name##_get_count(void){return name##_count;}
用途:统计关键函数调用次数(如中断处理、API调用):
COUNTED_FUNC(uart_send,int,constchar*data){
// 实际发送逻辑...
return0;
}
// 调用后可获取次数:
printf("UART发送次数: %d\n",uart_send_get_count());
避免副作用:表达式参数需用中间变量缓存(如MAX宏中的_a/_b),防止MAX(i++, j++)类问题。
括号保护:宏参数和整体需加括号,例如#define SQUARE(x) ((x)*(x)),避免SQUARE(1+2)展开为1+2*1+2=5的错误。
多行宏格式:用\连接多行,或用GCC扩展的({ ... })语句表达式(如MAX宏)。
类型安全:通过__typeof__(GCC扩展)或_Generic(C11)实现类型检查,例如:
#defineSAFE_ADD(a, b)_Generic((a),int:(a)+(b),float:(a)+(b))
这些宏函数体现了C语言"底层控制"与"代码复用"的平衡艺术——通过预处理阶段的文本替换,实现接近内联函数的性能,同时保持跨平台兼容性。实际开发中,应根据场景选择"极简宏"(如BIT_SET)或"安全宏"(如带类型检查的MAX),避免过度使用复杂宏导致代码可读性下降。最终目标是:让宏成为"隐形助手",而非"调试噩梦"。