您的位置:首页 > 编程语言 > C语言/C++

[C/C++]在头文件中使用static定义变量意味着什么

2017-06-30 09:21 465 查看
全局变量可不可以定义在可被多个.C文件包含的头文件中?

答案:可以,在不同的C文中中以static形式来声明同名的全局变量,可以在布偶年该的C文件中声明同名的全局变量,前提是其中只能有一个C文件中对此变量赋初值,此时连接不会出错!

建议不要在头文件中定义变量,头文件中只做变量的声明!

为什么要声明为static的原因如下:

1》每个.c文件,会编译成一个obj,这样,如果你在头文件中定义,而不是声明,那么每个引用这个头文件的obj中都会有一个这个变量的实例,连接的时候就会报重复定义出错了。

2》static变量的作用域只在该编译单元内有效

看到有一位同学在头文件中这么写:

1staticconst wchar_t* g_str1 = …
2staticconst wchar_t* g_str2 = …


这种定义变量的方式我从来没有见过,而且它还能顺利通过编译,于是我很想知道编译器是如何处理这种变量定义的。

定义全局变量时使用static,意味着该变量的作用域只限于定义它的源文件中,其它源文件不能访问。既然这种定义方式出现在头文件中,那么可以很自然地推测:包含了该头文件的所有源文件中都定义了这些变量,即该头文件被包含了多少次,这些变量就定义了多少次。

假如将上面两行代码的static去掉,编译的时候就会出现变量重定义的错误,这进一步证实了上面的推测,因为没有static的话变量的作用域是全局的,定义了两个以上的同名变量就会出现该错误。

推测终究是推测,要真正证实这个推测还要通过写代码来验证。验证的方式是:在头文件中使用static定义变量,在多个源文件中包含该头文件,然后在每个源文件中输出变量的地址,同时在一个源文件中改变变量的值并输出,在另一个源文件中也输出。如果每个源文件的输出都不同,则推测得证;否则推测是错误的。

下面是定义变量的头文件的代码:

1//Header.h2#pragma once
34staticint g_int = 3;


接下来在另一个头文件中声明两个测试函数:

1//Functions.h2#pragma once
34void TestSource1();
5void TestSource2();


分别在两个源文件中定义这两个测试函数:



1//Source1.cpp 2 #include  3 #include "Header.h" 4 5void TestSource1() {
6 7     wprintf(L"g_int's address in Source1.cpp: %08x\n", &g_int);
8     g_int = 5;
9     wprintf(L"g_int's value in Source1.cpp: %d\n", g_int);
10 }






1//Source2.cpp2 #include 3 #include "Header.h"45void TestSource2() {
67     wprintf(L"g_int's address in Source2.cpp: %08x\n", &g_int);
8     wprintf(L"g_int's value in Source2.cpp: %d\n", g_int);
9 }




最后在main函数中调用这两个测试函数:



1//Main.cpp2 #include "Functions.h"34int wmain() {
56    TestSource1();
7    TestSource2();
8 }




运行该程序:



 可以看到,虽然在代码中好像使用了相同的变量,但是实际上使用的是不同的变量,在每个源文件中都有单独的变量。所以,在头文件中定义static变量会造成变量多次定义,造成内存空间的浪费,而且也不是真正的全局变量。应该避免使用这种定义方式。

作为对比,下面使用正确的方式来定义全局变量:

1//Header.h2#pragma once
34externint g_int;




1//Source1.cpp 2 #include  3 #include "Header.h" 4 5int g_int = 3;
6 7void TestSource1() {
8 9     wprintf(L"g_int's address in Source1.cpp: %08x\n", &g_int);
10     g_int = 5;
11     wprintf(L"g_int's value in Source1.cpp: %d\n", g_int);
1213 }




其它文件不变。

运行程序:



 可以看到,这次两个源文件中使用的都是同一个变量。要注意的是,使用extern声明变量时不能带有初始值,否则仍然属于变量定义,会出现变量重定义的错误。

参考:点击打开链接
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐