编译单元(模块)
在VC或VS上编写完代码,点击编译按钮准备生成exe文件时,编译器做了两步工作:
- 将每个.cpp(.c)和相应的.h文件编译成obj文件;
- 将工程中所有的obj文件进行LINK,生成最终.exe文件。
那么,错误可能在两个地方产生:
- 编译时的错误,这个主要是语法错误;
-
链接时的错误,主要是
重复定义变量
等。
编译单元指在编译阶段生成的每个obj文件。
- 一个obj文件就是一个编译单元。
- 一个.cpp(.c)和它相应的.h文件共同组成了一个编译单元。
- 一个工程由很多编译单元组成,每个obj文件里包含了变量存储的相对地址等。
声明与定义
函数或变量在声明时,并没有给它实际的物理内存空间,它有时候可保证你的程序编译通过;
函数或变量在定义时,它就在内存中有了实际的物理空间。
如果你在编译单元中引用的外部变量没有在整个工程中任何一个地方定义的话,那么即使它在编译时可以通过,在连接时也会报错,因为程序在内存中找不到这个变量。
函数或变量可以声明多次,但定义只能有一次
。
extern作用
-
当它与”C”一起连用时,如
extern "C" void fun(int a, int b);
,则编译器在编译fun这个函数名时按C的规则去翻译相应的函数名而不是C++的。(C函数不能重载) -
当它不与”C”在一起修饰变量或函数时,如在头文件中,
extern int g_nNum;
,它的作用就是声明函数或变量的作用范围的关键字,其声明的函数和变量可以
在本编译单元或其他编译单元中
使用。
note:可以把全局变量的声明和定义放在一起,这样可以防止忘记了定义,如上面的
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中定义的做法
。
静态全局变量(static)
注意使用static修饰变量,就不能使用extern来修饰,即static和extern不可同时出现。
static修饰的全局变量的声明与定义同时进行,即当你在头文件中使用static声明了全局变量,同时它也被定义了
。
static修饰的全局变量的作用域只能是本身的编译单元。在其他编译单元使用它时,只是简单的把其值复制给了其他编译单元,其他编译单元会另外开个内存保存它,
在其他编译单元对它的修改并不影响本身在定义时的值
。即在其他编译单元A使用它时,它所在的物理地址,和其他编译单元B使用它时,它所在的物理地址不一样,A和B对它所做的修改都不能传递给对方。
多个地方引用静态全局变量所在的头文件,不会出现重定义错误,因为在每个编译单元都对它开辟了额外的空间进行存储。
全局常量(const)
const单独使用时,其特性与static一样(每个编译单元中地址都不一样,不过因为是常量,也不能修改,所以就没有多大关系)。
const与extern一起使用时,其特性与extern一样。
extern const char g_szBuffer[]; //写入 .h中
const char g_szBuffer[] = "123456"; // 写入.cpp中