资源预览内容
第1页 / 共6页
第2页 / 共6页
第3页 / 共6页
第4页 / 共6页
第5页 / 共6页
第6页 / 共6页
亲,该文档总共6页全部预览完了,如果喜欢就下载吧!
资源描述
全局变量、全局变量、extern/static/const 区别与联系区别与联系 在讨论全局变量之前我们先要明白几个基本的概念:1. 编译单元(模块):在 IDE 开发工具大行其道的今天,对于编译的一些概念很多人已经不再清楚了,很多程序员最怕的就是处理连接错误(LINK ERROR), 因为它不像编译错误那样可以给出你程序错误的具体位置,你常常对这种错误感到懊恼,但是如果你经常使用 gcc,makefile 等工具在 linux 或者嵌 入式下做开发工作的话,那么你可能非常的理解编译与连接的区别!当在 VC 这样的开发工具上编写完代码,点击编译按钮准备生成 exe 文件时,VC 其实做了两 步工作,第一步,将每个.cpp(.c)和相应.h 文件编译成 obj 文件;第二步,将工程中所有的 obj 文件进行 LINK 生成最终的.exe 文件,那么错 误就有可能在两个地方产生,一个是编译时的错误,这个主要是语法错误,另一个是连接错误,主要是重复定义变量等。我们所说的编译单元就是指在编译阶段生成 的每个 obj 文件,一个 obj 文件就是一个编译单元,也就是说一个一个 cpp(.c)和它相应的和它相应的.h 文件共同组成了一个编译单元,一个工程由很多个编译单元组文件共同组成了一个编译单元,一个工程由很多个编译单元组 成,每个成,每个 obj 文件里包含了文件里包含了变量存储的相对地址等变量存储的相对地址等 。2. 声明与定义的区别函数或变量在声明时,并没有给它实际的物理内存空间,函数或变量在声明时,并没有给它实际的物理内存空间,它有时候可以保证你的程序编译通过, 但是但是当函数或变量定义的时候,它就在内存中有了实际的物理空间,当函数或变量定义的时候,它就在内存中有了实际的物理空间,如果你在编译模块中引用的外部变量没有在整个工程中任何一个地方定义的话, 那么即使它在编译时可以通过,在连接时也会报错,因为程序在内存中找不到这个变量!你也可以这样理解, 对同一个变量或函数的声明可以有多次,而定义只能有一次!3. extern 的作用(见我的另外一篇文章总结见我的另外一篇文章总结)extern 有两个作用,第一个,当它与“C“一起连用时,如: extern “C“ void fun(int a, int b); 则告诉编译器在编译 fun 这个函数名时按着按着 C 的规则去翻译相应的函数名而不是的规则去翻译相应的函数名而不是 C+的的, C+的规则在翻译这个函数名时会把 fun 这个名字变得面目全非,可能是 funaBc_int_int#%$也可能是别的,这要看编译器的“脾气“了 (不同的编译器采用的方法不一样),为什么这么做呢,因为 C+支持函数的重载啊,在这里不去过多的论述这个问题,如果你有兴趣可以去网上搜索,相信你可 以得到满意的解释!当当 extern 不与不与“C“在一起修饰变量或函数时,如在头文件中在一起修饰变量或函数时,如在头文件中: extern int g_Int; 它的作用就是声它的作用就是声明函数或全局变量的作用范围的关键字,其声明的函数和变量可以在本模块或者其他模块中使用明函数或全局变量的作用范围的关键字,其声明的函数和变量可以在本模块或者其他模块中使用,记住它记住它是一个声明不是定义是一个声明不是定义!也就是说 B 模块(编译 单元)要是引用模块(编译单元)A 中定义的全局变量或函数时,它只要包含 A 模块的头文件即可, 在编译阶段,模块 B 虽然找不到该函数或变量,但它不会报错,它会在连接时从模块 A 生成的目标代码中找到此函数。如果你对以上几个概念已经非常明白的话,那么让我们一起来看以下几种全局变量/常量的使用区别:1. 用 extern 修饰的全局变量以上已经说了 extern 的作用,下面我们来举个例子,如: 在 test1.h 中有下列声明:#ifndef TEST1H#define TEST1Hextern char g_str; / 声明全局变量声明全局变量 g_strvoid fun1();#endif在 test1.cpp 中#include “test1.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 是整个工程的全局变量,在内存中只存在一份, 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 也包含了 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中声明的其他函数你也无法使用了,除非也用都用 extern 修饰,这样的话你光声明的函数就要一大串,而且头文件的作用就是要给外部提 供接口使用的,所以 请记住, 只在头文件中做声明,真理总是这么只在头文件中做声明,真理总是这么简单简单。2. 用 static 修饰的全局变量首先,我要告诉你我要告诉你 static 与与 extern 是一对是一对“水火不容水火不容”的家伙的家伙,也就是说 extern 和 static 不能同时修饰一个变量;其次,static 修修 饰的全局变量声明与定义同时进行饰的全局变量声明与定义同时进行,也就是说当你在头文件中使用static 声明了全局变量后,它也同时被定义了;最后,static 修饰全局变量的作用域修饰全局变量的作用域 只能是本身的编译只能是本身的编译单元单元,也就是说它的“全局”只对本编译单元有效,其他编译单元则看不到它,如: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 中找到它们,它们之所以可以连接成功而没有报重复定义的错误是因为虽然它们有相同的内容,但是存储的物理地址并不一样, 就像是两个不同变量赋了相同的值一样,而这两个变量这两个变量分别作用于它们各自的编译单元。分别作用于它们各自的编译单元。也许你比较较真,自己偷偷的跟踪调试上面的代码,结果你发现两个编译单元(test1, test2)的g_str 的内存地址相同,于是你下结论 static 修饰的变量也可以作用于其他模块,但是我要告诉你,那是你的编译器在欺骗你,大多数编 译器都对代码都有优化功能,以达到生成的目标程序更节省内存,执行效率更高,当编译器在连接各个编译单元的时候,它会把相同内容的内存只拷贝一份,比如上 面的“123456“, 位于两个编译单元中的变量都是同样的内容,那么在连接的时候它在内存中就只会存在一份了, 如果你把上面的代码改成下面的样子,你马上就可以拆穿编译器的谎言拆穿编译器的谎言: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 地址并不相同,因为你在一处修改了它,所以编译器被强行的恢复内存的原貌编译器被强行的恢复内存的原貌,在内存中存在了两份拷贝给两个模块中的变量使用。正是因为正是因为 static 有以上的特性,所以一般定义有以上的特性,所以一般定义 static 全局变量时,都把它放在原文件中而不是头文全局变量时,都把它放在原文件中而不是头文件,这样就不会给其他模块造成不必要的信息污染,同样记住这个原则吧!件,这样就不会给其他模块造成不必要的信息污染,同样记住这个原则吧!3 const 修饰的全局常量(const 总结的 ms 还不是很好,下次要是有好文章再转载!)const 修饰的全局常量用途很广,比如软件中的错误信息字符串都是用全局常量来定义的。const 修修饰的全局常量据有跟饰的全局常量据有跟 static 相同的特性相同的特性(有条件的,感谢有条件的,感谢 sswv 的提醒,的提醒,const 放在只读静态存储区放在只读静态存储区),即它们只能作用于本编译模块中,但是即它们只能作用于本编译模块中,但是 const 可以与可以与 extern 连用来声明该常量可以作用于其他编译模连用来声明该常量可以作用于其他编译模块中块中, 如extern const char g_str;然后在原文件中别忘了定义:const char g_str = “123456“;所以当 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
收藏 下载该资源
网站客服QQ:2055934822
金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号