(1)编译单元(模块)
在VC或VS上编写完代码,点击编译按钮准备生成exe文件时,编译器做了两步工作:
第一步,将每个.cpp(.c)和相应的.h文件编译成obj文件;
第二步,将工程中所有的obj文件进行LINK,生成最终.exe文件。
那么,错误可能在两个地方产生:
一个,编译时的错误,这个主要是语法错误;
一个,链接时的错误,主要是重复定义变量等。
编译单元指在编译阶段生成的每个obj文件。
一个obj文件就是一个编译单元。
一个.cpp(.c)和它相应的.h文件共同组成了一个编译单元。
一个工程由很多编译单元组成,每个obj文件里包含了变量存储的相对地址等。
(2)声明与定义
函数或变量在声明时,并没有给它实际的物理内存空间,它有时候可保证你的程序编译通过;
函数或变量在定义时,它就在内存中有了实际的物理空间。
如果你在编译单元中引用的外部变量没有在整个工程中任何一个地方定义的话,那么即使它在编译时可以通过,在连接时也会报错,因为程序在内存中找不到这个变量。
函数或变量可以声明多次,但定义只能有一次。
(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)
(5)静态全局变量(static)
有两个类都需要使用共同的变量,我们将这些变量定义为全局变量。比如,res.h和res.cpp分别来声明和定义全局变量,类ProducerThread和ConsumerThread来使用全局变量。(以下是QT工程代码)
/**********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使用全局变量************/ #include "consumerthread.h" #include "res.h" #include <QDebug> ConsumerThread::ConsumerThread(QObject* parent) : QThread(parent) { } ConsumerThread::ConsumerThread() { } ConsumerThread::~ConsumerThread() { } 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"; } /**************************/也可以把全局变量的声明和定义放在一起,这样可以防止忘记了定义,如上面的extern char g_szBuffer[g_nBufferSize]; 然后把引用它的文件中的#include "res.h"换成extern char g_szBuffer[];。
但是这样做很不好,因为你无法使用#include "res.h"(使用它,若达到两次及以上,就出现重定义错误;注:即使在res.h中加#pragma once,或#ifndef也会出现重复定义,因为每个编译单元是单独的,都会对它各自进行定义),那么res.h声明的其他函数或变量,你也就无法使用了,除非也都用extern修饰,这样太麻烦,所以还是推荐使用.h中声明,.cpp中定义的做法。
(5)静态全局变量(static)
注意使用static修饰变量,就不能使用extern来修饰,即static和extern不可同时出现。
static修饰的全局变量的声明与定义同时进行,即当你在头文件中使用static声明了全局变量,同时它也被定义了。
static修饰的全局变量的作用域只能是本身的编译单元。在其他编译单元使用它时,只是简单的把其值复制给了其他编译单元,其他编译单元会另外开个内存保存它,在其他编译单元对它的修改并不影响本身在定义时的值。即在其他编译单元A使用它时,它所在的物理地址,和其他编译单元B使用它时,它所在的物理地址不一样,A和B对它所做的修改都不能传递给对方。
多个地方引用静态全局变量所在的头文件,不会出现重定义错误,因为在每个编译单元都对它开辟了额外的空间进行存储。
以下是Windows控制台应用程序代码示例:
/***********res.h**********/ static char g_szBuffer[6] = "12345"; void fun(); /************************/
/***********res.cpp**********/ #include "res.h" #include <iostream> using namespace std; void fun() { for (int i = 0; i < 6; i++) { g_szBuffer[i] = 'A' + i; } cout<<g_szBuffer<<endl; } /************************/ /***********test1.h**********/ void fun1(); /************************/ /***********test1.cpp**********/ #include "test1.h" #include "res.h" #include <iostream> using namespace std; void fun1() { fun(); for (int i = 0; i < 6; i++) { g_szBuffer[i] = 'a' + i; } cout<<g_szBuffer<<endl; } /************************/ /***********test2.h**********/ void fun2(); /************************/ /***********test2.cpp**********/ #include "test2.h" #include "res.h" #include <iostream> using namespace std; void fun2() { cout<<g_szBuffer<<endl; } /************************/ /***********main.cpp**********/ #include "test1.h" #include "test2.h" int main() { fun1(); fun2(); system("PAUSE"); return 0; } /************************/
结果:
ABCDEF
abcdef
12345
按我们的直观印象,认为fun1()和fun2()输出的结果都为abcdef,可实际上fun2()输出的确是初始值。然后我们再跟踪调试,发现res、test1、test2中g_szBuffer的地址都不一样,分别为0x0041a020、0x0041a084、0x0041a040,这就解释了为什么不一样。
注:一般定义static 全局变量时,都把它放在.cpp文件中而不是.h文件中,这样就不会给其他编译单元造成不必要的信息污染。
(6)全局常量(const)
const单独使用时,其特性与static一样(每个编译单元中地址都不一样,不过因为是常量,也不能修改,所以就没有多大关系)。
const与extern一起使用时,其特性与extern一样。
extern const char g_szBuffer[]; //写入 .h中 const char g_szBuffer[] = "123456"; // 写入.cpp中
转自:http://blog.csdn.net/candyliuxj/article/details/7853938
相关推荐
宏定义、结构体定义、函数声明以及全局变量定义放到一个head.h头文件中 函数的定义放到head.cpp中 main函数放到main.cpp中 然而却报错了,提示xxx变量在*.obj文件中已定义 问题出现的原因 为什么会出现这种...
在C++编程语言中,全局变量是在程序的任何作用域之外声明的变量,可以在程序的多个函数或类中被访问。它们在整个程序的生命周期内都存在,从程序开始执行到结束时保持其值。全局变量的使用有其优点,如共享数据和...
C++中,全局变量只能声明、初始化,而不能赋值 也就是说,下面这样是不被允许的: #include using namespace std; int a; a = 2; int main() { return 0; } 错误提示是: C++ requires a type specifier for all...
#### 方法一:通过`.cpp`文件定义和引用全局变量 **定义**:首先在某个`.cpp`文件中定义全局变量,确保其定义位于任何函数之外。例如,在`a.cpp`中定义一个整型全局变量`int global;`。 **引用**:然后,在其他需要...
2. **声明全局变量**:在其他源文件中,我们需要使用`extern`关键字声明全局变量,表明它们是在其他文件中定义的,例如: ```cpp // main.cpp #include "global_variables.h" // 包含全局变量的头文件 int main() {...
总之,C++的函数声明和定义以及变量声明和定义是程序编译和链接过程中的重要概念,它们保证了编译器和链接器能够正确处理代码中的函数调用和跨文件的变量引用。了解这些概念及其工作原理对于编写高效、可维护的C++...
全局变量是在所有函数之外定义的变量,具有全局作用域。这意味着一旦在某个源文件中定义,它可以被整个程序中的所有源文件访问。为了确保其他源文件能够识别并使用这个变量,通常需要使用`extern`关键字进行声明。 ...
然而,合理地利用`extern`和全局变量的声明与定义,可以帮助构建跨越多个编译单元的复杂系统。在实践中,更推荐使用静态成员变量或局部静态变量,以及智能指针来管理共享资源,以降低全局状态的负面影响。同时,良好...
在MFC(Microsoft Foundation Classes)框架下开发应用时,全局变量和全局函数的定义与普通C++项目有所不同,因为MFC项目的结构更为复杂,通常由多个文件构成,直接在类外部定义全局变量可能会导致编译错误或者链接...
全局变量是在函数外部定义的变量,可以在程序的任何地方被访问。在多线程环境中,如果一个全局变量被多个线程同时读写,由于处理器调度的不确定性,可能导致数据不一致性和错误的结果。这种现象通常称为竞态条件,是...
1、声明全局变量 2、声明函数 今天我们只谈extern,什么const、static之类等等与之相关或不相关的一律忽略,下面就分别对以上两种情况一一讲解 声明和定义 既然提到extern声明变量,那我们就必须搞清楚声明和定义的...
常规C++全局变量 这是最基础的方法,适用于不涉及MFC特定类型的全局变量。在MFC应用的任何源文件中(通常是`.cpp`文件),你可以在全局作用域定义一个变量: ```cpp extern int g_GlobalInt; // 声明 int g_...
`extern` 关键字在C/C++语言中用于声明全局变量或者外部变量。当你在一个源文件中定义了一个全局变量,而在其他源文件中想要使用这个变量时,由于编译器默认每个源文件都是独立的,所以它不知道这个变量的存在,这时...
### MFC中定义全局变量和全局函数的方法 #### 引言 在Microsoft Foundation Classes (MFC)框架下进行软件开发时,经常会遇到需要在不同类之间共享数据或功能的情况。由于MFC项目的复杂性,简单地定义全局变量或函数...
### C/C++中静态全局变量、静态局部变量、全局变量及局部变量的深入解析 #### 一、概念区分 在C/C++编程语言中,**静态全局变量**、**静态局部变量**、**全局变量**及**局部变量**是经常使用的几种变量类型,它们...
它们根据其定义的位置和特性,可以分为几种不同的类型:局部变量、全局变量、局部静态变量和全局静态变量。这些变量各有其特点和作用域,理解它们之间的区别对于编写高效且无误的C++代码至关重要。 1. 局部变量...
全局变量在编程中是一种特殊...最后,`1.cpp`文件很可能是实现上述描述的C++源代码,它可能包含了`count`、`temp`、`func1`和`func2`的具体定义和实现。通过阅读和分析这个文件,可以更深入地理解这些概念的实际应用。
一个,链接时的错误,主要是重复定义变量等。 编译单元指在编译阶段生成的每个obj文件。 一个obj文件是一个编译单元。 一个.cpp(.c)和它相应的.h文件共同组成了一个编译单元。 一个工程由很多编译...