如何定义全局变量

上传人:mg****85 文档编号:35885293 上传时间:2018-03-22 格式:DOC 页数:5 大小:31KB
返回 下载 相关 举报
如何定义全局变量_第1页
第1页 / 共5页
如何定义全局变量_第2页
第2页 / 共5页
如何定义全局变量_第3页
第3页 / 共5页
如何定义全局变量_第4页
第4页 / 共5页
如何定义全局变量_第5页
第5页 / 共5页
亲,该文档总共5页,全部预览完了,如果喜欢就下载吧!
资源描述

《如何定义全局变量》由会员分享,可在线阅读,更多相关《如何定义全局变量(5页珍藏版)》请在金锄头文库上搜索。

1、定义 C/C+全局变量/常量几种方法的区别(转)C/C+技术 2009-10-05 20:26:45 阅读293 评论0 字号:大中小 在讨论全局变量之前我们先要明白几个基本的概念:1. 编译单元(模块):在 IDE 开发工具大行其道的今天,对于编译的一些概念很多人已经不再清楚了,很多程序员最怕的就是处理连接错误(LINK ERROR), 因为它不像编译错误那样可以给出你程序错误的具体位置,你常常对这种错误感到懊恼,但是如果你经常使用 gcc,makefile 等工具在 linux 或者嵌入式下做开发工作的话,那么你可能非常的理解编译与连接的区别!当在 VC 这样的开发工具上编写完代码,点击编

2、译按钮准备生成 exe 文件时,VC 其实做了两步工作,第一步,将每个.cpp(.c)和相应.h 文件编译成 obj 文件;第二步,将工程中所有的 obj 文件进行 LINK 生成最终的.exe 文件,那么错误就有可能在两个地方产生,一个是编译时的错误,这个主要是语法错误,另一个是连接错误,主要是重复定义变量等。我们所说的编译单元就是指在编译阶段生成的每个 obj 文件,一个 obj 文件就是一个编译单元,也就是说一个 cpp(.c)和它相应的.h 文件共同组成了一个编译单元,一个工程由很多个编译单元组成,每个 obj 文件里包含了变量存储的相对地址等 。2. 声明与定义的区别函数或变量在声明

3、时,并没有给它实际的物理内存空间,它有时候可以保证你的程序编译通过, 但是当函数或变量定义的时候,它就在内存中有了实际的物理空间,如果你在编译模块中引用的外部变量没有在整个工程中任何一个地方定义的话, 那么即使它在编译时可以通过,在连接时也会报错,因为程序在内存中找不到这个变量!你也可以这样理解,对同一个变量或函数的声明可以有多次,而定义只能有一次!3. extern 的作用extern 有两个作用,第一个,当它与“C“一起连用时,如: extern “C“ void fun(int a, int b); 则告诉编译器在编译 fun 这个函数名时按着 C 的规则去翻译相应的函数名而不是 C+的

4、, C+的规则在翻译这个函数名时会把 fun 这个名字变得面目全非,可能是funaBc_int_int#%$也可能是别的,这要看编译器的“脾气“了(不同的编译器采用的方法不一样),为什么这么做呢,因为 C+支持函数的重载啊,在这里不去过多的论述这个问题,如果你有兴趣可以去网上搜索,相信你可以得到满意的解释!当 extern 不与“C“在一起修饰变量或函数时,如在头文件中: extern int g_Int; 它的作用就是声明函数或全局变量的作用范围的关键字,其声明的函数和变量可以在本模块活其他模块中使用,记住它是一个声明不是定义!也就是说 B 模块(编译单元)要是引用模块(编译单元)A 中定义

5、的全局变量或函数时,它只要包含 A 模块的头文件即可, 在编译阶段,模块 B 虽然找不到该函数或变量,但它不会报错,它会在连接时从模块 A 生成的目标代码中找到此函数。如果你对以上几个概念已经非常明白的话,那么让我们一起来看以下几种全局变量/常量的使用区别:1. 用 extern 修饰的全局变量以上已经说了 extern 的作用,下面我们来举个例子,如: 在 test1.h 中有下列声明:#ifndef TEST1H#define TEST1Hextern char g_str; / 声明全局变量 g_strvoid fun1();#endif在 test1.cpp 中#include “te

6、st1.h“char g_str = “123456“; / 定义全局变量 g_strvoid fun1()cout g_str endl;以上是 test1模块, 它的编译和连接都可以通过,如果我们还有 test2模块也想使用g_str,只需要在原文件中引用就可以了#include “test1.h“void fun2()cout g_str endl;以上 test1和 test2可以同时编译连接通过,如果你感兴趣的话可以用 ultraEdit 打开test1.obj,以你可在里面着“123456“这个字符串,但是你却不能在 test2.obj 里面找到,这是因为 g_str 是整个工程的

7、全局变量,在内存中只存在一份, test2.obj 这个编译单元不需要再有一份了,不然会在连接时报告重复定义这个错误!有些人喜欢把全局变量的声明和定义放在一起,这样可以防止忘记了定义,如把上面test1.h 改为extern char g_str = “123456“; / 这个时候相当于没有 extern然后把 test1.cpp 中的 g_str 的定义去掉,这个时候再编译连接 test1和 test2两个模块时,会报连接错误,这是因为你把全局变量 g_str 的定义放在了头文件之后,test1.cpp这个模块包含了 test1.h 所以定义了一次 g_str,而 test2.cpp 也包

8、含了 test1.h 所以再一次定义了 g_str, 这个时候连接器在连接 test1和 test2时发现两个 g_str。如果你非要把g_str 的定义放在 test1.h 中的话,那么就把 test2的代码中#include “test1.h“去掉 换成:extern char g_str;void fun2()cout g_str endl;这个时候编译器就知道 g_str 是引自于外部的一个编译模块了,不会在本模块中再重复定义一个出来,但是我想说这样做非常糟糕,因为你由于无法在 test2.cpp 中使用#include “test1.h“, 那么 test1.h 中声明的其他函数你也

9、无法使用了,除非也用都用extern 修饰,这样的话你光声明的函数就要一大串,而且头文件的作用就是要给外部提供接口使用的,所以 请记住, 只在头文件中做声明,真理总是这么简单。2. 用 static 修饰的全局变量首先,我要告诉你 static 与 extern 是一对“水火不容”的家伙,也就是说 extern和 static 不能同时修饰一个变量;其次,static 修饰的全局变量声明与定义同时进行,也就是说当你在头文件中使用 static 声明了全局变量后,它也同时被定义了;最后,static 修饰全局变量的作用域只能是本身的编译单元,也就是说它的“全局”只对本编译单元有效,其他编译单元则

10、看不到它,如:test1.h:#ifndef TEST1H#define TEST1Hstatic char g_str = “123456“; void fun1();#endiftest1.cpp:#include “test1.h“void fun1()cout g_str endl;test2.cpp#include “test1.h“; void fun2()cout g_str endl;以上两个编译单元可以连接成功, 当你打开 test1.obj 时,你可以在它里面找到字符串“123456“, 同时你也可以在 test2.obj 中找到它们,它们之所以可以连接成功而没有报重复定义

11、的错误是因为虽然它们有相同的内容,但是存储的物理地址并不一样,就像是两个不同变量赋了相同的值一样,而这两个变量分别作用于它们各自的编译单元。也许你比较较真,自己偷偷的跟踪调试上面的代码,结果你发现两个编译单元(test1, test2)的 g_str 的内存地址相同,于是你下结论 static 修饰的变量也可以作用于其他模块,但是我要告诉你,那是你的编译器在欺骗你,大多数编译器都对代码都有优化功能,以达到生成的目标程序更节省内存,执行效率更高,当编译器在连接各个编译单元的时候,它会把相同内容的内存只拷贝一份,比如上面的“123456“, 位于两个编译单元中的变量都是同样的内容,那么在连接的时候

12、它在内存中就只会存在一份了, 如果你把上面的代码改成下面的样子,你马上就可以拆穿编译器的谎言:test1.cpp:#include “test1.h“void fun1()g_str0 = a;cout g_str endl;test2.cpp#include “test1.h“void fun2()cout g_str endl;void main()fun1(); / a23456fun2(); / 123456这个时候你在跟踪代码时,就会发现两个编译单元中的 g_str 地址并不相同,因为你在一处修改了它,所以编译器被强行的恢复内存的原貌,在内存中存在了两份拷贝给两个模块中的变量使用。正

13、是因为 static 有以上的特性,所以一般定义 static 全局变量时,都把它放在原文件中而不是头文件,这样就不会给其他模块造成不必要的信息污染,同样记住这个原则吧!3 const 修饰的全局常量const 修饰的全局常量用途很广,比如软件中的错误信息字符串都是用全局常量来定义的。const 修饰的全局常量据有跟 static 相同的特性,即它们只能作用于本编译模块中,但是 const 可以与 extern 连用来声明该常量可以作用于其他编译模块中, 如extern const char g_str;然后在原文件中别忘了定义:const char g_str = “123456“;所以当

14、const 单独使用时它就与 static 相同,而当与 extern 一起合作的时候,它的特性就跟 extern 的一样了!所以对 const 我没有什么可以过多的描述,我只是想提醒你,const char* g_str = “123456“ 与 const char g_str = “123465“是不同的, 前面那个 const 修饰的是 char * 而不是 g_str,它的 g_str 并不是常量,它被看做是一个定义了的全局变量(可以被其他编译单元使用) , 所以如果你像让 char *g_str 遵守 const 的全局常量的规则,最好这么定义 const char* const g_str=“123456“.

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 生活休闲 > 科普知识

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号