![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=1739066369-R7c2ADxV2tT7k1qI85UwrfgJqR49UwjO-0-d715e00fddca034c748f30817741e821)
预处理器无论在什么时候遇到了这样的指令,任何出现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=1739066369-UtMRQo7LX12RC2TXY3dgrdbAoTgZaZQJ-0-0f57bbcb62fcc2428c116501618a8888)
【代码详解】
在该例中,使用宏预处理器定义了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=1739066369-eL7QzvkZAuSZj9qdIKH6AJsRbdF5lbuB-0-25b90d962b5ac91d0a48a556b94eb974)
图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=1739066369-NASNFxQKop8UsvdZcbT0JC2g7IZQHzdG-0-9f307bc7bc5fae823e68b93792592958)
像122这样的数字在程序中被称为魔法数字。一个魔法数字是hard-coded数字,它在代码中没有任何意义—122表示什么呢?是转换率还是其他什么呢?它是不明确的。在一些复杂的程序里,通常很难判断一个hard-coded数字具体代表什么。
下面的小段代码是清晰的:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15768.jpg?sign=1739066369-k9KYng3H7cTQRhdAKv2O0FaFVzBJUkzF-0-84f8ab105a0f50db4da139ca99c19c69)
其次,#defined数字可以使得程序更加容易被修改。假设将转换率从122变成123,程序需要进行相应的调整。考虑下面的代码:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15769.jpg?sign=1739066369-RucpbYK47MmuYRqS5QLm7H6V0LEXVNV3-0-4644fc58c051deb8f7842175246085fe)
为了改变成新的转换率,必须将前面4个语句中的数字改变。但是第5个语句呢?这里的122是不是和其他的122具有相同意义呢?如果是,它就应该被改变;如果不是,就不需要改变,或者也许在其他地方中断。
现在考虑使用了#defined的情况,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15770.jpg?sign=1739066369-LLcFLTVym1pYar9vdMKTYAlNvR7Yh97A-0-1183c85f7ffaa8fd6f476c2faf917191)
这时改变转换率只要改变一个数字,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15771.jpg?sign=1739066369-wIPRuxGzP9KqSof3g4sAeuDYbRViUlNR-0-30ea7fd205e6fa25783f0bb454dbd8bf)
现在正确改变了转换率,并且不用担心将每页的行数改变。
6.5.3 const修饰符
常类型是指使用类型修饰符const说明的类型。常类型的变量或对象的值不能被更新。
提示
编译器通常不为普通const常量分配存储空间,而是将它们保存在符号表中,这使得它成为一个编译期间的常量,没有了存储与读内存的操作,使得它的效率也很高。因此,定义或说明常类型时必须进行初始化。
1.一般常量
一般常量是指简单类型的常量。这种常量在定义时,修饰符const可以用在类型说明符前,也可以用在类型说明符后,例如:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15870.jpg?sign=1739066369-Lg8L1YCqePXZk20Nni59AGZkL4gxcWKM-0-82051f9a9443f4ec46aaa6cc2b85b795)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15871.jpg?sign=1739066369-s4VfIG2oVkdQEyeipwNuk3PxCGLypz0s-0-ad5c4dfca86afb451f9eb3e10a94d48c)
定义或说明一个常数组可采用如下格式:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15872.jpg?sign=1739066369-9tVjtLbPYEjelBcG5gaD1yqGOxS2Gf3M-0-7868a3f88a743c23019ff392cbb4cba2)
2.常对象
常对象是指对象常量,定义格式如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15873.jpg?sign=1739066369-fxHl9sOQswsL1g31IL3N8gfQ9ByVwGOV-0-321417b72198c09d38a73d6269033510)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15874.jpg?sign=1739066369-5wArRWgD0etFlnAmQeeQqmAJuU7ITULh-0-f713945a3716d190a8a7aec64e76bf0d)
定义常对象时,同样要进行初始化,并且该对象不能再被更新,修饰符const可以放在类名后面,也可以放在类名前面。