5. 关于类的声明和定义。
class A; //类的声明
类的声明和普通变量声明一样,不产生目标代码,可以在同一,以及多个编译单元重复声明。
class A {
}; //类的定义 www.2cto.com
类的定义就特殊一点了,可能会有疑问,为什么不能把int x;这样的变量定义放到.h中(见4)但是可以把
类的定义放在头文件中重复引用呢?同时类的函数非inline定义(写在类定义里面的函数是inline,除外)不能写在
头文件中呢。
这是因为类的定义,只是告诉编译器,类的数据格式是如何的,实例话后对象该占多大空间。
类的定义也不产生目标代码。因此它和普通变量的声明唯一的区别是不能在同一编译单元内出现多次。
//source1.cc
class A;
class A; //类重复声明,OK
class A{
};
class A{
};
class A{
int x;
}; //同一编译单元内,类重复定义,会编译时报错,因为编译器不知道在该编译单元,A a;的话要生产怎样的a.
//如果class A{};定义在head.h ,而head.h 没有
//#ifndef #endif 就很可能在同一编译单元出现类重复定义的编译错误情况。
但是在不同编译单元内,类可以重复定义,因为类的定义未产生实际代码。
//source1.cc
class A{
}
//source2.cc
class A{
} //不同编译单元,类重复定义,OK。所以类的定义可以写在头文件中!
//source1.cc
class A{
}
//source2.cc
class A{
int x;
} //不同编译单元,OK
6. 总结
1.在头文件中写变量的声明,函数声明,类的定义,inline函数,不要出现变量定义,类的函数非inline定义,函数定
义。
即在头文件中不要出现可能产生目标代码的东东。
2.为了防止在一个编译单元内部头文件重复引用,所有头文件都要加上#ifndef #endif
3.鼓励在.cc中使用不具名namespace,可以有效防止不同编译单元命名冲突。
4.相关更专业详细的介绍可以看<<大规模C++程序设计>>的第一章,会有极其好的完整介绍。
其中提到类的定义是具有内部链接特性的,即它不是声明
不能在同一编译单元重复出现,但是它具有内部链接,(所谓内部链接指的是该名称对于所在编译单元是局部的,在链接时不会与其他编译单元中同样
的名称产生命名冲突),所以类如果要在单个编译单元之外使用它必须被定义在一个头文件中。
对于声明和定义,书中给出的定义是:
一个声明将一个名称引入程序,一个定义提供了一个实体(例如,类型,实例,函数)在一个程序中的唯一描述。
5. 前面第一条说的不是很确切,按照<<大规模C++程序设计>>中的说法
理论上头文件中可以放所有具有内部链接的东西,包括具有内部链接的定义。如
static int x;
static void 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
还是很常用的啊。难道也要在
.h中
extern const int width;
.cc中
const int width = 5;
这样虽然可以,不过好麻烦啊,我倒觉得在.h中定义类似const int width =3 问题不大,难道编译器不会做些特殊的处理优化吗,也要每个单元分配一个单独空间?
不过倒是也可以利用下面的方法在.h中声明一批const 变量。注意和普通static 变量不同,类的成员静态变量,静态函数是具有外部链接的。如果
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_code INVALID_ADDRESS = 1;//wrong addres
static const result_code READ_FAIL = 2;//cannot read
static const result_code WRITE_FAIL = 3;//cannot write
static const result_code UNKNOWN_ACTION = 4;//dunno...
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
};
摘自 XpowerLord