类可以定义在函数体内,但这样的类毕竟受到一些限制。所以,类一般都不定义在函数体内。当在函数体外定义,在各个指定的源文件中可能只有一处该类的定义。而且,如果要在不同文件中使用同一个类,类的定义就必须保持一致。
为了确保各个文件中类的定义一致,类通常被定义在头文件中,而且类所在的头文件名应该和类名一致。例如:库类型string在名为string的头文件中定义。头文件通常包含那些只能被定义一次的实体,如类、const和consetexpr变量等。头文件也经常用到其他头文件的功能。例如,一个Sales_data类中用到string,所以sales_data.h必须包含string.h头文件。同时在包含Sales_data类的文件中用到string时包含了string.h头文件。这样的情况下,两次包含了string.h头文件
确保头文件多次包含仍能正常工作的技术是预处理器。预处理器是在编译之前执行的一段程序,如#include,当预处理器看到#include标记时就会用指定的头文件内容代替#include。C++还会用到一项预处理功能是头文件保护符,头文件保护符依赖于预处理变量,预处理变量有两种状态:已定义和未定义。#define指令把一个名字设定为预处理变量,另外两个则分别检查某个预处理变量是否已经定义:#ifdef当且仅当变量已经定义为真,#ifndef当且仅当未定义时为真。一旦检查结果为真,则执行后续操作直到遇到**#endif**为止。
使用这些功能可以有效地防止重复包含的发生:
#ifndef SALES_DATA_H
#define SALES_DATA_H
#include <string>
struct Sales_data
{
std::string bookNo;
unsigned units_sold = 0;
double revenue = 0.0;
}
#endif
第一次包含Sales_data.h时,#ifndef检查为真,预处理器将顺序执行后面的操作直到遇到#endif为止。此时,预处理变量SALES_DATA_H的值将变为已定义,而且Sales_data.h也会被拷贝到我们的程序中,后面如果再一次包含Sales_data.h则#ifndef检查为假,编译器将忽视#ifndef和#endif之间的部分。
头文件保护符很简单,程序员只要习惯的加上就可以,没必要太在乎程序需不需要。