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

C语言专题:15.宏定义与控制指令(#define、#ifndef、#undef、#defined)

         C语言的预处理阶段为我们提供了灵活而强大的宏机制,其中 #define 用于定义宏常量与宏函数,配合 #ifndef#undefdefined 可以实现条件编译、头文件保护、功能开关、调试控制等。


一、#define:定义宏

1.1 宏常量

#define PI 3.14159
#define MAX_LEN 100
  • 所有后续代码中出现 PI 都会被预处理器替换成 3.14159

  • 宏没有类型限制,完全是文本替换


1.2 宏函数(带参数)

#define SQUARE(x) ((x) * (x))

示例:

printf("%d\n", SQUARE(5)); // 输出 25

注意:

  • 必须使用括号保护参数和整个表达式;

  • 宏没有类型检查,不推荐写过于复杂的表达式。


1.3 多行宏定义

#define LOG(msg)         \printf("[LOG] %s\n", msg); \fflush(stdout);

使用 \ 表示换行宏,便于格式化长逻辑块。


二、#ifndef:判断未定义宏

2.1 基本语法

#ifndef 宏名// 如果未定义这个宏,执行这部分
#endif

常用于头文件保护

// file: mylib.h
#ifndef MYLIB_H
#define MYLIB_H// 声明内容...#endif

目的:防止多次包含头文件造成重定义错误


2.2 使用场景

场景示例
头文件保护#ifndef MY_H ... #endif
防止重复定义#ifndef MAX_SIZE #define MAX_SIZE 100
功能模块开关#ifndef DISABLE_SOUND

三、#undef:取消宏定义

#define DEBUG
#undef DEBUG
  • 取消宏定义后,#ifdef DEBUG 条件将返回 false;

  • 可用于临时关闭某些功能或在不同区块重新定义。


四、defined 运算符(用于条件判断)

4.1 基本语法

#if defined(DEBUG)printf("Debug Mode\n");
#endif

也可用于逻辑判断:

#if defined(DEBUG) && !defined(NLOG)printf("Debug and Logging\n");
#endif
  • defined 是一个预处理器内置运算符

  • 只能用于 #if/#elif 表达式中。


4.2 推荐写法对比

写法说明
#ifdef A宏 A 被定义
#if defined(A)更通用,适用于复杂表达式
#if !defined(A)等价于 #ifndef A

五、宏机制工作原理示意

#define MAX 100int arr[MAX];

预处理后等价于:

int arr[100];
  • 宏处理发生在编译前的预处理阶段

  • 宏不会出现在目标文件(.o / .exe)中;

  • 仅是文本层面的替换


六、实战:条件编译 + 宏定义联合应用

#define WINDOWS
#define ENABLE_LOG#if defined(WINDOWS)#define PLATFORM_NAME "Windows"
#else#define PLATFORM_NAME "Other"
#endif#ifdef ENABLE_LOG#define LOG(msg) printf("[LOG] %s\n", msg)
#else#define LOG(msg) // 空宏,不执行
#endifLOG(PLATFORM_NAME);  // 输出: [LOG] Windows

七、宏指令 vs const vs inline

功能#defineconstinline
类型检查
是否占用内存不占编译器决定
调试查看不可调试可查看可查看
替代时机编译前(预处理)编译时编译/链接时
推荐使用场景控制编译结构常量表达式小函数、性能优化

八、注意事项与建议

问题说明
宏函数缺括号#define SQR(x) x*x,会导致错误计算
宏名冲突尽量加前缀,如 MYLIB_MAX_SIZE
重定义宏建议先 #undef 后重新定义
宏调试困难使用 const 替代复杂值更方便调试
滥用宏建议宏只用于控制编译、常量表达式、调试开关等

九、小结一览表

指令功能说明示例
#define定义宏常量或宏函数#define PI 3.14
#undef取消已定义的宏#undef DEBUG
#ifndef如果宏未定义,则编译对应代码#ifndef MYLIB_H
defined条件判断中测试宏是否定义#if defined(WIN32)

十、结语

  • #define 是 C 语言最基础的预处理机制,适用于常量定义与条件控制;

  • 搭配 #ifdef/#ifndef/#undef/#defined 可以实现灵活的编译开关管理

  • 宏强大但容易滥用,推荐将复杂逻辑迁移到 constinline 函数中;

  • 写宏时一定要注意括号和命名,避免产生预处理陷阱。

http://www.lqws.cn/news/536131.html

相关文章:

  • MySQL(基础篇)
  • [特殊字符] Windows 查看端口占用及服务来源教程(以 9018 端口为例)
  • Oracle LogMiner分析日志的三种方法示例
  • UDP 和 TCP 可以同时使用相同的端口号
  • el-table表头添加说明
  • Excel基础:数据编辑
  • Excel:filter函数实现动态筛选的方法
  • 网络分层模型与协议体系技术研究报告
  • 微信小程序<rich-text>支持里面图片点击放大
  • 物联网与低代码:Node-RED如何赋能工业智能化与纵横智控的创新实践
  • 【51单片机5毫秒定时器】2022-6-1
  • 机器学习---正则化、过拟合抑制与特征筛选
  • 抖音图文带货权限怎么开通
  • Vue3 中 Axios 深度整合指南:从基础到高级实践引言总结
  • 《解锁FFmpeg - python:开启多媒体处理新时代》
  • 多线程编程 ----线程主动退出pthread_exit与线程被动退出pthread_cancel
  • DAY 33 简单的神经网络
  • 前端面试专栏-主流框架:14. Vue Router与Vuex核心应用
  • Spring Boot使用Redis常用场景
  • Python爬虫多线程并发时的503错误处理最佳实践
  • HTTP-Cookie和Session
  • 算法第48天|单调栈:42. 接雨水、84.柱状图中最大的矩形
  • 鸿蒙边缘智能计算架构实战:从环境部署到分布式推理全流程
  • window显示驱动开发—DirectX 图形内核子系统(一)
  • 树莓派超全系列教程文档--(67)rpicam-apps可用选项介绍之检测选项
  • 算法-最大子数组
  • 【Python】For
  • Agentic AI爆发前夜,合作伙伴如何把握时代机遇?
  • 2D写实交互数字人如何重塑服务体验?
  • MP1652GTF-Z:MPS高效3A降压转换器 工业5G通信专用