【C语言】全局搜索变量却找不到定义?原来是因为宏!

简介: 使用条件编译和 `extern` 来管理全局变量的定义和声明是一种有效的技术,但应谨慎使用。在可能的情况下,应该优先考虑使用局部变量、函数参数和返回值、静态变量或者更高级的封装技术(如结构体和类)来减少全局变量的使用。

使用条件编译和 extern 关键字来管理全局变量的定义和声明

在开发过程中,我们经常需要在多个源文件之间共享全局变量。为了避免重复定义的错误,同时确保变量的唯一性和可访问性,可以使用条件编译和 extern 关键字来管理变量的定义和声明。下面将详细介绍这种技术,并通过示例代码说明其应用。

1. 头文件 module.h

#ifndef MODULE_H
#define MODULE_H

#undef EXTERN // 取消之前可能存在的 EXTERN 宏的定义

#ifdef _MODULE_C_ // 检查是否定义了 _MODULE_C_ 宏
#define EXTERN // 如果定义了,则定义 EXTERN 为空(即不添加任何修饰符)
#else
#define EXTERN extern // 如果没有定义,则定义 EXTERN 为 extern
#endif

// 使用 EXTERN 宏来声明或定义 globalVar 变量
EXTERN volatile int globalVar;

#endif /* MODULE_H */

2. 源文件 module.c

#define _MODULE_C_ // 定义宏,表示当前是定义变量的源文件

#include "module.h"

// 这里不需要再次定义 globalVar,因为在 module.h 中已经处理了变量的定义

void some_function() {
   
    // 使用 globalVar 进行操作
    globalVar = 1; // 示例操作
}

3. 其他源文件

在其他源文件中,只需要包含 module.h 头文件,由于它们没有定义 _MODULE_C_ 宏,EXTERN 将被定义为 extern,因此它们将只包含 globalVar 变量的声明,而不是定义。

#include "module.h"

// 在这里使用 globalVar,由于 EXTERN 被定义为 extern,所以不会重复定义变量
void another_function() {
   
    // 使用 globalVar 进行操作
    globalVar = 2; // 示例操作
}

4. 代码详解

4.1. module.h

module.h 头文件中,通过条件编译来决定 EXTERN 的定义。如果 _MODULE_C_ 宏被定义,那么 EXTERN 将为空,表示在当前源文件中进行变量定义;否则,EXTERN 被定义为 extern,表示在其他源文件中进行变量声明。

#undef EXTERN // 取消之前可能存在的 EXTERN 宏的定义

#ifdef _MODULE_C_ // 检查是否定义了 _MODULE_C_ 宏
#define EXTERN // 如果定义了,则定义 EXTERN 为空(即不添加任何修饰符)
#else
#define EXTERN extern // 如果没有定义,则定义 EXTERN 为 extern
#endif

// 使用 EXTERN 宏来声明或定义 globalVar 变量
EXTERN volatile int globalVar;

4.2. module.c

module.c 源文件中,定义 _MODULE_C_ 宏,然后包含 module.h。由于在 module.h 中,EXTERN 被定义为空,所以 globalVarmodule.c 中被定义。注意,这里不需要再次显式定义 globalVar,因为在 module.h 中已经处理了变量的定义。

#define _MODULE_C_ // 定义宏,表示当前是定义变量的源文件

#include "module.h"

// 这里不需要再次定义 globalVar,因为在 module.h 中已经处理了变量的定义

void some_function() {
   
    // 使用 globalVar 进行操作
    globalVar = 1; // 示例操作
}

4.3. 其他源文件

在其他源文件中,只需要包含 module.h 头文件,由于它们没有定义 _MODULE_C_ 宏,EXTERN 将被定义为 extern,所以 globalVar 在这些源文件中只是声明而不是定义。

#include "module.h"

// 在这里使用 globalVar,由于 EXTERN 被定义为 extern,所以不会重复定义变量
void another_function() {
   
    // 使用 globalVar 进行操作
    globalVar = 2; // 示例操作
}

5. 优缺点分析

5.1. 优点

  • 避免多重定义错误:通过条件编译和 extern 关键字,可以确保全局变量只在一个源文件中定义,在其他源文件中声明,避免了链接时的多重定义错误。
  • 提高代码可维护性:通过宏和条件编译,可以灵活控制变量的定义和声明,提高代码的可维护性和可读性。
  • 模块化编程:可以实现代码的模块化,将变量的定义和使用分开,使得代码结构更清晰。

5.2. 缺点

  • 增加复杂性:使用宏和条件编译会增加代码的复杂性,可能导致理解和维护困难。
  • 隐式依赖:代码依赖于特定的宏定义,可能导致隐式依赖,不容易追踪变量的定义和声明位置。
  • 调试困难:在调试过程中,可能难以确定变量的定义和声明位置,增加调试难度。

5.3. 结论

使用条件编译和 extern 来管理全局变量的定义和声明是一种有效的技术,但应谨慎使用。在可能的情况下,应该优先考虑使用局部变量、函数参数和返回值、静态变量或者更高级的封装技术(如结构体和类)来减少全局变量的使用。

通过这种方法,可以确保 globalVar 在整个工程中只有一个定义,同时在需要的地方都可以访问到这个变量,避免了多重定义错误。

6. 结束语

  1. 本节内容已经全部介绍完毕,希望通过这篇文章,大家对条件编译和 extern 来管理全局变量有了更深入的理解和认识。
  2. 感谢各位的阅读和支持,如果觉得这篇文章对你有帮助,请不要吝惜你的点赞和评论,这对我们非常重要。再次感谢大家的关注和支持
目录
相关文章
|
5月前
|
C语言
C语言15---在VC中变量的注意点和变量的注意细节
C语言15---在VC中变量的注意点和变量的注意细节
|
6月前
|
Java 程序员 Linux
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
探索C语言宝库:从基础到进阶的干货知识(类型变量+条件循环+函数模块+指针+内存+文件)
49 0
|
7月前
|
程序员 编译器 C语言
C语言进阶⑳(程序环境和预处理)(#define定义宏+编译+文件包含)(中)
C语言进阶⑳(程序环境和预处理)(#define定义宏+编译+文件包含)
41 0
|
7月前
|
自然语言处理 编译器 Linux
C语言进阶⑳(程序环境和预处理)(#define定义宏+编译+文件包含)(下)
C语言进阶⑳(程序环境和预处理)(#define定义宏+编译+文件包含)
66 0
|
7月前
|
存储 程序员 编译器
C语言进阶⑳(程序环境和预处理)(#define定义宏+编译+文件包含)(上)
C语言进阶⑳(程序环境和预处理)(#define定义宏+编译+文件包含)
53 0
|
7月前
|
C语言
C语言多文件编译、结构体、枚举及联合
C语言多文件编译、结构体、枚举及联合
47 0
|
存储 自然语言处理 编译器
程序编译和链接的过程/预处理符号和用法【C语言】
程序编译和链接的过程/预处理符号和用法【C语言】
74 0
|
Serverless C语言
C语言——利用函数实现某一特定功能
C语言——利用函数实现某一特定功能
309 0
|
Java 编译器 C语言
学C的第四天(各种操作符补充;简单了解多种关键字,define定义常量和宏,结构体)(2)
[被static修饰的变量(静态变量)和全局变量的区别:都存放在静态区中,但静态变量的作用域还是在局部,全局变量的作用域是任意地方,两者生命周期相同,都是程序结束才销毁] 2.修饰全局变量 - 称为静态全局变量 (未使用static时:)
|
C语言
学C的第四天(各种操作符补充;简单了解多种关键字,define定义常量和宏,结构体)(1)
10.6:关系操作符: >:大于 >=:大于等于 <:小于 <=:小于等于 !=:用于测试“不相等” ==:用于测试“相等”