![C++从零开始学(视频教学版)(第2版)](https://wfqqreader-1252317822.image.myqcloud.com/cover/128/29977128/b_29977128.jpg)
6.5 预处理器
预处理器是一个独立的程序,在编译器编译程序之前运行。虽然它们不是C++的一部分,但是却扩展了C++程序设计的环境。这样做的目的是处理指令,这些指令是以#符号开始的,独立占用一行,不能使用分号结束。本节将介绍其中的一种,就是宏预处理器#define。
6.5.1 #define预处理器
#define是宏定义命令,宏定义具有这样的形式:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P121_15632.jpg?sign=1739006745-4Zj0so2mx5eoprD6ds2tH4nY9C1xNyiC-0-b36c390019d5885e499cdc55232bd6cf)
预处理器无论在什么时候遇到了这样的指令,任何出现identifier的地方都将被替换成replacement。标识符通常为大写字母,使用下画线代替空格。
提示
在写多行的代码define时,最好在外层加上do{}while(0),效率不会影响,并且避免在不加{}的if中使用宏的错误。
通过一个实例来说明#define如何使用。
【实例6-10】define的使用(代码6-10.txt)
新建名为“definetest”的【C++ Source File】源程序,源代码如下所示:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P121_15633.jpg?sign=1739006745-bzdMsNpLEuH8hzd6YfEXQqLepnODyiGG-0-acff14da5dcfb93021ac65949b05c0d4)
【代码详解】
在该例中,使用宏预处理器定义了YEN_PER_DOLLAR为122;在主程序中,首先定义int型变量i并赋值为5,接下来i赋值为i*宏名,将i的结果输出。
运行结果如图6-11所示。
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P121_15629.jpg?sign=1739006745-HWpbAgb713vOnphaNglU0vhjSv2TOi15-0-72d284f18b2d01f215250eefec0ec359)
图6-11 代码运行结果
【实例分析】
从运行结果来看,输出i的结果就是122*5的结果。这里YEN_PER_DOLLAR看起来像一个变量,但它与变量没有任何关系,它只是一个符号或标志,在程序代码编译前,此符号会用122来代替。122不是一个数值,只是一个字符串,不会进行检查。
6.5.2 #define的作用
通过6.5.1节的介绍认识了#define预处理器,那么为什么要引入这个预处理器呢?首先,允许给一些东西命名为描述性的名字,如数字。
举个例子:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15767.jpg?sign=1739006745-93WkVAOMKzzb4P530Z3iQt2wAvgtVo1k-0-f49aab2dd29de7cf687bef6e6a8376db)
像122这样的数字在程序中被称为魔法数字。一个魔法数字是hard-coded数字,它在代码中没有任何意义—122表示什么呢?是转换率还是其他什么呢?它是不明确的。在一些复杂的程序里,通常很难判断一个hard-coded数字具体代表什么。
下面的小段代码是清晰的:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15768.jpg?sign=1739006745-fFtFgvFMLkKNAG18uNO1Ul0XqLQYgFk7-0-c531909233a08c44a5f214048bb4d7a6)
其次,#defined数字可以使得程序更加容易被修改。假设将转换率从122变成123,程序需要进行相应的调整。考虑下面的代码:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15769.jpg?sign=1739006745-P91Rvd7hA0chXSIGsULi96C28v6p92F8-0-32f8c87d34c98dbcd9ba8b30adec37ac)
为了改变成新的转换率,必须将前面4个语句中的数字改变。但是第5个语句呢?这里的122是不是和其他的122具有相同意义呢?如果是,它就应该被改变;如果不是,就不需要改变,或者也许在其他地方中断。
现在考虑使用了#defined的情况,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15770.jpg?sign=1739006745-AhHtriT9FKboX8G7nKLzEBOKVJ2bXHmG-0-c7fba3f1cc835a00d6cc73559e4fb9e8)
这时改变转换率只要改变一个数字,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15771.jpg?sign=1739006745-vJZ7Gz3RKUh0LMMO46HyGLb6ubtQfE3G-0-58498e2efed065b6927f818edb2e0515)
现在正确改变了转换率,并且不用担心将每页的行数改变。
6.5.3 const修饰符
常类型是指使用类型修饰符const说明的类型。常类型的变量或对象的值不能被更新。
提示
编译器通常不为普通const常量分配存储空间,而是将它们保存在符号表中,这使得它成为一个编译期间的常量,没有了存储与读内存的操作,使得它的效率也很高。因此,定义或说明常类型时必须进行初始化。
1.一般常量
一般常量是指简单类型的常量。这种常量在定义时,修饰符const可以用在类型说明符前,也可以用在类型说明符后,例如:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15870.jpg?sign=1739006745-W1Pa7u7D8nRIlTw99lfo3Vsbbw3nHPSc-0-24a95233e4ed0094921a3cf1b9d66bcd)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15871.jpg?sign=1739006745-7B9tpsfSfjpxI2xvx1TDtkRqHTcdsgqi-0-774bc08b1db247160b465ca3ebfe14da)
定义或说明一个常数组可采用如下格式:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15872.jpg?sign=1739006745-eQAQ6C292FDVaDiUAU6MAuVCc7PE3yJH-0-7f6fb04a3fb7a2c1dcb5b28fc1137723)
2.常对象
常对象是指对象常量,定义格式如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15873.jpg?sign=1739006745-LqFaHXPiDlTXLGbrR4pB9GO0wztyhUni-0-ddb41d04f2f09199c9c490860915965d)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15874.jpg?sign=1739006745-CjOmZXugLYkPDXfUHm0FhfAoUkJOifFo-0-c310dae7dc6732c9d65a6563c0965995)
定义常对象时,同样要进行初始化,并且该对象不能再被更新,修饰符const可以放在类名后面,也可以放在类名前面。