全局变量的声明、定义及用法

文章目录

  • 全局变量的声明、定义及用法
    • 1. 编译单元(模块)
    • 2. 声明和定义
    • 3. extern 作用
    • 4. 全局变量(extern)
      • 4.1 如果直接将声明和定义都放在头文件中会如何?
    • 5. 静态全局变量(static)
    • 6. 全局常量(const)

转载自https://blog.csdn.net/candyliuxj/article/details/7853938
转载自https://blog.csdn.net/chaipp0607/article/details/59110710

2019年2月27日 15:27:44修改

1. 编译单元(模块)

在 VC 或 VS 上编写完代码,点击编译按钮准备生成 exe 文件时,编译器做了两步工作:

  • 将每个 .cpp(.c) 和相应的. h 文件编译成 obj 文件;
  • 将工程中所有的 obj 文件进行 LINK,生成最终 .exe 文件。

那么,错误可能在两个地方产生:

  • 编译时的错误,这个主要是语法错误;
  • 链接时的错误,主要是重复定义变量等。

编译单元指在编译阶段生成的每个 obj 文件。

一个 obj 文件就是一个编译单元。
一个 .cpp(.c) 和它相应的 .h 文件共同组成了一个编译单元。
一个工程由很多编译单元组成,每个 obj 文件里包含了变量存储的相对地址等。

2. 声明和定义

函数或变量在声明时,并没有给它实际的物理内存空间,它有时候可保证你的程序编译通过;函数或变量在定义时,它就在内存中有了实际的物理空间。

如果你在编译单元中引用的外部变量没有在整个工程中任何一个地方定义的话,那么即使它在编译时可以通过,在链接时也会报错,因为程序在内存中找不到这个变量。

函数或变量可以声明多次 (如果是变量多次声明需要加 extern 关键字),但定义只能有一次。

对于变量而言,在同一个文件里面,很少使用声明这个说法,一般没有人去说我要声明一个变量,然后定义这个变量。或者说,同一文件下声明与定义没有明显的区别。就好比 int a;我们可以说这是一个声明,也可以说这是一个定义,因为当程序执行到这句话的时候就完成了内存分配。数据类型,变量名,对应的内存单元就已经明确。
而 int a =1; 就完成了初始化,因为它明确了内存单元里到底存放什么样的数据。

在同一个工程,在多个文件中变量的声明和定义才有区别。比如说在 first.c 文件中先定义了一个全局变量 int a; 我在 second.c 中要访问这个 a ,这时我们需要在 second.c 这个文件中用 extern 声明一下。这是常规写法,我们有更好的写法,在下面介绍。

3. extern 作用

  • 当它与 “C” 一起连用时,如 extern “C” void fun(int a, int b);,则编译器在编译 fun 这个函数名时按 C 的规则去翻译相应的函数名而不是 C++ 的。

  • 当它不与 “C” 在一起修饰变量或函数时,如在头文件中,extern int g_nNum;,它的作用就是声明函数或变量的作用范围的关键字,其声明的函数和变量可以在本编译单元或其他编译单元中使用。

即 B 编译单元要引用 A 编译单元中定义的全局变量或函数时,B 编译单元只要包含 A 编译单元的头文件即可,在编译阶段,B 编译单元虽然找不到该函数或变量,但它不会报错,它会在链接时从 A 编译单元生成的目标代码中找到此函数。

4. 全局变量(extern)

有两个以上文件都需要使用共同的变量,我们将这些变量定义为全局变量。比如,res.h 和 res.cpp 分别来 声明定义 全局变量,其他文件需要使用全局变量时,只需要包含 res.h 即可,同时这也是 推荐的写法

下面是示例代码,只为演示全局变量的用法,不保证算法完备性:

/**********res.h声明全局变量************/
#pragma once#include <QSemaphore>const int g_nDataSize = 1000; // 生产者生产的总数据量
const int g_nBufferSize = 500; // 环形缓冲区的大小extern char g_szBuffer[]; // 环形缓冲区
extern QSemaphore g_qsemFreeBytes; // 控制环形缓冲区的空闲区(指生产者还没填充数据的区域,或者消费者已经读取过的区域)
extern QSemaphore g_qsemUsedBytes; // 控制环形缓冲区中的使用区(指生产者已填充数据,但消费者没有读取的区域)
/**************************/

上述代码中g_nDataSize、g_nBufferSize为全局常量,其他为全局变量。

/**********res.cpp定义全局变量************/
#pragma once
#include "res.h"// 定义全局变量
char g_szBuffer[g_nBufferSize];
QSemaphore g_qsemFreeBytes(g_nBufferSize);
QSemaphore g_qsemUsedBytes;
/**************************/

在其他编译单元中使用全局变量时只要包含其所在头文件即可。

/**********在consumerthread.cpp使用全局变量************/
#include "consumerthread.h"
#include "res.h"
#include <QDebug>void ConsumerThread::run() {for (int i = 0; i < g_nDataSize; i++) {g_qsemUsedBytes.acquire();              qDebug()<<"Consumer "<<g_szBuffer[i % g_nBufferSize];g_szBuffer[i % g_nBufferSize] = ' ';g_qsemFreeBytes.release();}qDebug()<<"&&Consumer Over";
}
/**************************/

为了程序的简洁和清爽,我们通常将变量声明放在头文件中,需要使用的时候只需要包含头文件即可,如果不这样做,当涉及到的全局变量变多时,我们要做一大堆繁琐的声明工作。

4.1 如果直接将声明和定义都放在头文件中会如何?

效果看起来是这样:

/**********res.h声明全局变量************/
#pragma once#include <QSemaphore>const int g_nDataSize = 1000;
const int g_nBufferSize = 500; char g_szBuffer[];
QSemaphore g_qsemFreeBytes;
QSemaphore g_qsemUsedBytes;
/**************************/

这样做非常不妙,如果你的工程简单到只有一个头文件和一个源文件,那么没问题。但是当工程变得复杂时,很容易出现重定义错误。

看到其他的说法是,即使在 res.h 中加 #pragma once,或 #ifndef 也会出现重复定义,因为每个编译单元是单独的,都会对它各自进行定义),那么 res.h 声明的其他函数或变量,你也就无法使用了,除非也都 extern 修饰,这样太麻烦,所以还是推荐使用 .h 中声明,.cpp 中定义的做法

但是我个人的验证是 #pragma once,或 #ifndef 可以有效防止重定义的问题,遇到使用 #ifndef 也会提示重定义的情况是 #ifndef 仅对代码片段有效,在 #ifndef 范围外的代码自然会提示重定义。

这里先保留意见,总之,.h中声明,.cpp中定义的做法是最稳妥的。

5. 静态全局变量(static)

注意使用 static 修饰变量,就不能使用 extern 来修饰,即 static 和 extern 不可同时出现。

static 修饰的全局变量的声明与定义同时进行,即当你在使用 static 声明了全局变量,同时它也被定义了。

static 修饰的全局变量的作用域只能是本身的编译单元。在其他编译单元使用它时,只是简单的把其值复制给了其他编译单元,其他编译单元会另外开个内存保存它,在其他编译单元对它的修改并不影响本身在定义时的值。即在其他编译单元 A 使用它时,它所在的物理地址,和其他编译单元 B 使用它时,它所在的物理地址不一样,A 和 B 对它所做的修改都不能传递给对方。

多个地方引用静态全局变量所在的头文件,不会出现重定义错误,因为在每个编译单元都对它开辟了额外的空间进行存储。

下面是示例程序,同样只做演示,不保证算法完备性:

/***********res.h**********/
static char g_szBuffer[6] = "12345";
/************************//***********test1.cpp**********/
#include "res.h"//fun1对g_szBuffer中的内容进行修改
void fun1() { for (int i = 0; i < 6; i++) {g_szBuffer[i] = 'a' + i;}cout<<g_szBuffer<<endl;
}
/************************//***********test2.cpp**********/
#include "res.h"//fun2对g_szBuffer原样打印
void fun2() {cout<<g_szBuffer<<endl;
}
/************************//***********main.cpp**********/
#include "test1.h"
#include "test2.h"int main() {fun1();fun2();system("PAUSE");return 0;
}
/************************/

按我们的直观印象,认为 fun1() 和 fun2() 输出的结果都为相同,可实际上 fun2() 输出的确是初始值。然后我们再跟踪调试,发现 test1、test2 中 g_szBuffer 的地址都不一样,这就解释了为什么不一样。

注:一般定义 static 全局变量时,都把它放在 .cpp 文件中而不是 .h 文件中,这样就不会给其他编译单元造成不必要的信息污染。

6. 全局常量(const)

const单独使用时,其特性与static一样(每个编译单元中地址都不一样,不过因为是常量,也不能修改,所以就没有多大关系)。

const 与 extern 一起使用时,其特性与 extern 一样。

extern const char g_szBuffer[];      //写入 .h中
const char g_szBuffer[] = "123456"; // 写入.cpp中

全局变量的声明、定义及用法相关推荐

  1. 基于C++全局变量的声明与定义的详解

    (1)编译单元(模块) 在VC或VS上编写完代码,点击编译按钮准备生成exe文件时,编译器做了两步工作: 第一步,将每个.cpp(.c)和相应的.h文件编译成obj文件: 第二步,将工程中所有的obj ...

  2. C++全局变量的声明和定义

    参考:http://wrchen.blog.sohu.com/71617539.html (1)编译单元(模块)     在VC或VS上编写完代码,点击编译按钮准备生成exe文件时,编译器做了两步工作 ...

  3. php全局变量的关键字,PHP变量作用域(全局变量局部变量)globalstatic关键字用法实例分析...

    本文实例讲述了PHP变量作用域(全局变量&局部变量)&global&static关键字用法.分享给大家供大家参考,具体如下: 我们知道,变量呢,其实就相当于我们用来储存信息的容 ...

  4. python中全局变量和局部变量的区别_Python全局变量与局部变量区别及用法分析

    本文实例讲述了Python全局变量与局部变量区别及用法.分享给大家供大家参考,具体如下: 对于很多初学的同学,对全局和局部变量容易混淆,看看下面给大家的讲解相信都应该明白两者的区别了. 定义: 全局变 ...

  5. android object数组赋值_Java对象数组定义与用法详解

    本文实例讲述了Java对象数组定义与用法.分享给大家供大家参考,具体如下: 所谓的对象数组,就是指包含了一组相关的对象,但是在对象数组的使用中一定要清楚一点:数组一定要先开辟空间,但是因为其是引用数据 ...

  6. C++的this指针【定义、用法、本质、特点】

    一.this指针的定义及用法 我们知道在C++中成员变量和成员函数是分开存储的,每一个非静态成员函数只会诞生一份函数实例,也就是说多个同类型的对象会共用一块代码. 那么问题是:这一块代码是如何区分哪个 ...

  7. java多态讲解例子_Java多态性定义与用法实例详解

    本文实例讲述了Java多态性定义与用法.分享给大家供大家参考,具体如下: 多态性是通过: 1 接口和实现接口并覆盖接口中同一方法的几不同的类体现的 2 父类和继承父类并覆盖父类中同一方法的几个不同子类 ...

  8. java 定义抽象变量_Java抽象类和抽象方法定义与用法实例详解

    本文实例讲述了Java抽象类和抽象方法定义与用法.分享给大家供大家参考,具体如下: 一.Java抽象类 1.抽象类的说明 在面向对象的概念中,所有的对象都是通过类来描绘的,但是反过来,并不是所有的类都 ...

  9. 函数指针及其定义和用法

    函数指针及其定义和用法 1.什么是函数指针 如果在程序中定义了一个函数,那么在编译时系统就会为这个函数代码分配一段存储空间,这段存储空间的首地址称为这个函数的地址.而且函数名表示的就是这个地址.既然是 ...

最新文章

  1. 预留创建时检查增强点:nbsp;MB_RE…
  2. EasyUI——实现展示后台数据代码
  3. linux入门_Linux从入门到入土(抽奖送10本新书)
  4. mysql数据设置浮动_浮动float
  5. ArcGIS for qml -添加自由文本
  6. 写在自己工作六年:转载《软件工程师六年心得体会》
  7. BP算法的matlab代码学习
  8. nginx日志采集 mysql_shell + go + mysql nginx日志统计 (三) :数据的展示
  9. Python argv小结
  10. Windows下解压tar.gz压缩文件
  11. 适合入门的linux教程,Linux入门记录系列教程,适合Linux初学者阅读
  12. VS创建和使用C++动态链接库教程
  13. 联想e550笔记本怎么样_摄像头是亮点 — Lenovo 联想 ThinkPad E550C 笔记本 简单评测...
  14. 极速office(word)如何在方框内打钩
  15. django restframework serializer 增加自定义字段
  16. 最全的人力资源行业精美报表模板,免费下载啦
  17. 计算机网络工程用排线架,网络配线架使用和安装说明【图解】
  18. app兼容性测试方案
  19. 动态规划 资源分配问题
  20. 【Android - 技术期刊】第002期

热门文章

  1. 从ODX(Open diagnostic Data eXchange)谈车联网应用绕不开的底层网络“基建”
  2. 计算机黑屏启动超慢,电脑启动黑屏,要等待很久才能进入操作界面?为什么?
  3. 【Three.js入门】渲染第一个场景及物体(轨道控制器、坐标轴辅助器、移动缩放旋转)
  4. Activity 页面跳转,打招呼
  5. 【按键精灵源码分享】遍历图片在屏幕上出现的所有坐标脚本源码
  6. 项目管理软件应该具备哪些功能
  7. 饥荒游侠服务器未响应,游侠开饥荒服务器老掉线 | 手游网游页游攻略大全
  8. 一维消消乐c语言数据结构,Python数据结构--一维开心消消乐
  9. ios录音文件路径_导出iPhone6录音文件两种方法可搞定!-手机录音在哪个文件夹...
  10. 讓15個成功人物搖頭的壞習慣 [转]