《2022年C++关于声明,定义,类的定义,头文件作用 .pdf》由会员分享,可在线阅读,更多相关《2022年C++关于声明,定义,类的定义,头文件作用 .pdf(11页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、C+ 关于声明,定义,类的定义,头文件作用,防止头文件在同一个编译单元重复引用,不具名空间1. 编译单元,一个 .cc ,或.cpp 作为一个编译单元 .生成.o2. 普通数据类型的定义,声明,函数的定义声明(类函数是一样的)extern int x; /变量是声明,并未实际分配地址,未产生实际目标代码void print(); / 函数声明,未产生实际目标代码如 int x; int x = 3 ; void print() ; /均为定义 产生了实际目标代码 。声明不产生实际的目标代码,它的作用是告诉编译器,OK,我在该编译单元后面,或者其它编译单元会有这个x 变量, print函数的定义
2、。否则编译器如果发现程序用到x,print ,而前面没有声明会报错。如果有声明,而没有定义,那么链接的时候会报错未定义。比较常见的是我在source.cc中调用 print(),而 head.h中声明print (),而 source.cc 中 includehead.h从而就有了 print的声明,可以通过编译,但是如果在所有编译单元中没有 print函数的定义,那么链接的时候 source.o单元就会出错,因为它试图用print函数但是找不到print的定义。/head.hvoid pirnt();/source.ccvoid foo() print();由于声明不产生实际代码,所以可以有
3、多个重复声明的存在。名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 1 页,共 11 页 - - - - - - - - - /source1.ccextern int x;/source2.ccextern int x;甚至同一个编译单元也可以有多各个重复声明/source1.ccextern int x;extern int x;而普通变量定义,函数定义是不允许的。3. 同一编译单元内部的重名符号在编译期就被阻止了,而不同编译单元之间的重名符号要到链接器才会被发现。如果你在一个s
4、ource1.cc中/source1.ccint x;int x;出现两次int x; int x;即两个 x 的定义,会编译报错 ,x 重复定义。如果你的/source1.ccint x;/source2.ccint x;g+ o test source1.cc source2.cc那么编译过程不会出错,在链接过程,由于目标代码中有两个全局域的x,会链接出错, x 重定义。不同的编程人员可能会写不同的模块,那么很容易出现这种情况, 如何避免呢,namespace可以避免重名。google 编程规范鼓励使用 不具名空间名师资料总结 - - -精品资料欢迎下载 - - - - - - - - -
5、 - - - - - - - - - 名师精心整理 - - - - - - - 第 2 页,共 11 页 - - - - - - - - - /source1.ccnamespace int x;/source2.ccnamespace int x;OK, 现在不会链接出错了因为两个x 不重名了,当然对于这个简单的例子只在 source1.cc中用不具名命名空间就可避免链接出差了。 / 注 /source1.ccnamespace int x; /source1.cc static int x;有什么区别呢,看上去效果一样,区别在于不具名空间的x 仍然具有外链接,但是由于它是不具名的,所以别的
6、单元没办法链接到,如果namespace hahaint x; 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 3 页,共 11 页 - - - - - - - - - 则在别的单元可以用haha:x访问到它, static 则因为是内部链接特性,所以无法链接到。C+ 中 static 和 anonymousenamespace的差别2009-01-02 14:54 | 分类: 桌面应用开发记得以前一个同事问我为什么程序里使用了anonymousenamespace, 想了想就回答说
7、其实就是保持局部性(这也是我的目的),然后就有人说为什么不用static ,嗯 似乎这 两 个 东 西 乍 一 看 没 什 么 区 别 , 自 己 便Google了 一 下 , 发 现 有 一 个 原 因 就 是anonymousenamespace 里的member 都是有外部链接的,只不过永远都不能被外部link到!而 static就明确为根本没有外部链接!此时就出现问题了,在模板里无类型的参数必须是有外部链接的才可以,否则编译无法通;比如:template class Foobar ; namespace void abc() wcout_T( ”abc”)endl; static vo
8、id efg() wcout_T( ”efg ”)endl; int _tmain(int argc, _TCHAR* argv) Foobarxyz /! ;这一行可以通过Foobarrst; /! 注意这一行编译不过return 0; 也有人认为使用anon namespace比较好,因为static的方式被C+98 标准所批评,呵呵 总体来说,其实你完全可以用anony namespace代替 static 。4. 关于头文件。/head.hint x;/source1.cc#include “ head.h ”名师资料总结 - - -精品资料欢迎下载 - - - - - - - - -
9、 - - - - - - - - - 名师精心整理 - - - - - - - 第 4 页,共 11 页 - - - - - - - - - /source2.cc#include “ head.h ”头文件不被编译, .cc 中的引用include “ head.h”其实就是在预编译的时候将 head.h中的内容插入到 .cc 中。所以上面的例子如果g+ o test source1.cc source2.cc, 同样会链时发现重复定义的全局变量 x。因此变量定义, 包括函数的定义不要写到头文件中,因为头文件很可能要被多个.cc 引用。那么如果我的 head.h如下这么写呢,是否防止了x 的
10、链接时重定义出错呢?/head.h#ifndef _HEAD_H_#define _HEAD_H_int x;#endif/source1.cc#include “ head.h ”/source2.cc#include “ head.h ”现在是否 g+ o test source1.cc source2.cc就没有问题了呢,答案是否定的。所有的头文件都是应该如上加#ifndef #endif的, 但它的作用是防止头文件在 同一编译单元 被重复引用。就是说防止可能的/source1.cc#include “ head.h ”#include “ head.h ”名师资料总结 - - -精品资
11、料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 5 页,共 11 页 - - - - - - - - - 这种情况,当然我们不会主动写成上面的形式但是,下面的情况很可能发送/source1.cc#include “ head.h ”#inlcude “ a.h ”/a.h#include “ head.h ”这样就在不经意见产生了同一编译单元的头文件重复引用,于是 soruc1.cc 就出现了两个 int x; 定义。但是对于不同的编译单元source1.cc,source2.cc他们都是还会引用head.h的,
12、即使 #ifndef #endif的存在。5. 关于类的声明和定义。class A; / 类的声明类的声明和普通变量声明一样,不产生目标代码,可以在同一,以及多个编译单元重复声明。class A ;/ 类的定义类的定义就特殊一点了,可能会有疑问,为什么不能把int x; 这样的变量定义放到 .h 中(见 4)但是可以把类的定义放在头文件中重复引用呢?同时类的函数非inline 定义( 写在类定义里面的函数是 inline ,除外)不能写在头文件中呢。这是因为类的定义,只是告诉编译器,类的数据格式是如何的,实例话后对象该占多大空间。类的定义也不产生目标代码。 因此它和普通变量的声明唯一的区别是不
13、能在同一编译单元内出现多次。/source1.ccclass A;class A; / 类重复声明, OK名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 6 页,共 11 页 - - - - - - - - - class A;class A; class Aint x; / 同一编译单元内,类重复定义,会编译时报错,因为编译器不知道在该编译单元, A a ;的话要生产怎样的a./ 如果 class A;定义在 head.h ,而 head.h 没有/#ifndef #endif 就
14、很可能在同一编译单元出现类重复定义的编译错误情况。但是在不同编译单元内,类可以重复定义,因为类的定义未产生实际代码。/source1.ccclass A/source2.ccclass A / 不同编译单元,类重复定义,OK。所以类的定义可以写在头文件中!/source1.ccclass A/source2.ccclass Aint x;名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 7 页,共 11 页 - - - - - - - - - / 不同编译单元 ,OK6.总结1. 在头
15、文件中写变量的声明,函数声明,类的定义,inline 函数,不要出现变量定义,类的函数非inline定义,函数定义。即在头文件中不要出现可能产生目标代码的东东。2. 为了防止在一个编译单元内部头文件重复引用,所有头文件都要加上#ifndef #endif3. 鼓励在 .cc 中使用不具名 namespace,可以有效防止不同编译单元命名冲突。4. 相关更专业详细的介绍可以看 的第一章,会有极其好的完整介绍。其中提到 类的定义是具有内部链接特性的,即它不是声明不能在同一编译单元重复出现,但是它具有内部链接,(所谓内部链接指的是该名称对于所在编译单元是局部的, 在链接时不会与其他编译单元中同样的名
16、称产生命名冲突),所以类如果要在单个编译单元之外使用它必须被定义在一个头文件中。对于声明和定义,书中给出的定义是:一个声明将一个名称引入程序,一个定义提供了一个实体(例如,类型,实例,函数)在一个程序中的唯一描述。 5. 前面第一条说的不是很确切,按照 中的说法名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 8 页,共 11 页 - - - - - - - - - 理论上头文件中可以放所有具有内部链接的东西,包括具有内部链接的定义 。如 static int x; static vo
17、id print() ;但是 不提倡这么做 ,因为每一个包含这个头文件的.cc 就对应要开辟一个空间存储这个x, 就是说不同编译单元都引入static int x;由于是内部链接,所以互不影响彼此。甚至你采用 namespace也是如此,如在.h 中namespace myspace static int x;不同 .cc 文件中都引入该头文件,在各自编译单元中调用的myspace:x也是不同的互不影响的!书中提到 const int width = 3; / 见书的 23 页这样的 const变量也要避免出现在头文件中,不过类似以前c 语言中在头文件中 #define width 3名师资料
18、总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 9 页,共 11 页 - - - - - - - - - 还是很常用的啊。难道也要在 .h 中 extern const int width; .cc 中const int width = 5;这样虽然可以,不过好麻烦啊,我倒觉得在.h 中定义类似const int width =3 问题不大,难道编译器不会做些特殊的处理优化吗,也要每个单元分配一个单独空间?不过倒是也可以利用下面的方法在.h 中声明一批 const 变量。注意和普通stati
19、c 变量不同,类的成员静态变量,静态函数是具有外部链接的。如果static const int SUCCESS = 0; ,SUCCESS不是 const 仅仅是static int,那么是不可以在类内初始化的(编译出错),需要在某个 .cc文件中初始话,因为它是具有外部链接的。( 在 GOOGLE 编程规范中,提到禁止使用类类型的全局变量,静态成员变量视为全局变量,也禁止使用类类型) class code public: static const result_code SUCCESS = 0;/program ended successfully static const result_c
20、ode INVALID_ADDRESS = 1;/wrong addres static const result_code READ_FAIL = 2;/cannot read 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 10 页,共 11 页 - - - - - - - - - static const result_code WRITE_FAIL = 3;/cannot write static const result_code UNKNOWN_ACTION = 4;/d
21、unno. static const result_code NOT_FOUND = 5;/key not found in paragraph static const result_code NO_WRITE = 6;/no write since modification static const result_code SYNTAX_ERR = 7;/command syntax error static const result_code EMPTY_CLIP = 8;/the clipboard is empty ; 名师资料总结 - - -精品资料欢迎下载 - - - - - - - - - - - - - - - - - - 名师精心整理 - - - - - - - 第 11 页,共 11 页 - - - - - - - - -