![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=1739035479-hczntYD9bzdMRlzKiE9xAVRaYLvvX4wi-0-c21ec808994c89e04d2b14baf25899c7)
预处理器无论在什么时候遇到了这样的指令,任何出现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=1739035479-4nd23pQXVxCkusNoCJI85NkZd4XXKwwb-0-eac1aa61b98ceec9633de8fe4342efbd)
【代码详解】
在该例中,使用宏预处理器定义了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=1739035479-qymJ0AdjIn2cEeWVo78dEbwMgXZDFDgN-0-d5eb9517dd40e27beb6912121602b711)
图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=1739035479-Qdruu1Bsfu0qvke8Y7mWqxXOBteCfovo-0-9515bf43a6fd19558efa7b3d0d64ed86)
像122这样的数字在程序中被称为魔法数字。一个魔法数字是hard-coded数字,它在代码中没有任何意义—122表示什么呢?是转换率还是其他什么呢?它是不明确的。在一些复杂的程序里,通常很难判断一个hard-coded数字具体代表什么。
下面的小段代码是清晰的:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15768.jpg?sign=1739035479-H1pZRcdoqMfRcQdGgdnYpqJ3indCbpYp-0-b3b587454fe70aa23db88b2915bd669e)
其次,#defined数字可以使得程序更加容易被修改。假设将转换率从122变成123,程序需要进行相应的调整。考虑下面的代码:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15769.jpg?sign=1739035479-63coyy5Z53G9ctruDfu9vxdkPjMXHv8n-0-11e1270a570829661810987339eb1532)
为了改变成新的转换率,必须将前面4个语句中的数字改变。但是第5个语句呢?这里的122是不是和其他的122具有相同意义呢?如果是,它就应该被改变;如果不是,就不需要改变,或者也许在其他地方中断。
现在考虑使用了#defined的情况,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15770.jpg?sign=1739035479-tpcXxP3SAhWW65ayWqb21n9OJ0MH83Na-0-3746f602f8b735565e18f449363b627f)
这时改变转换率只要改变一个数字,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15771.jpg?sign=1739035479-E8Rg0mTtg0WqqakLqwqlt0cSEeBJBvwE-0-99e4b0b34468fbb674d28e7f62bba1ef)
现在正确改变了转换率,并且不用担心将每页的行数改变。
6.5.3 const修饰符
常类型是指使用类型修饰符const说明的类型。常类型的变量或对象的值不能被更新。
提示
编译器通常不为普通const常量分配存储空间,而是将它们保存在符号表中,这使得它成为一个编译期间的常量,没有了存储与读内存的操作,使得它的效率也很高。因此,定义或说明常类型时必须进行初始化。
1.一般常量
一般常量是指简单类型的常量。这种常量在定义时,修饰符const可以用在类型说明符前,也可以用在类型说明符后,例如:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15870.jpg?sign=1739035479-TZ3WyfpH7z3YoG8QJOc9WczqvweUqWbI-0-ba34491e1409c4eececa3dd73aa3ad14)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15871.jpg?sign=1739035479-O5H4z9bsson8Dv1FzqkQLPk9L5ldB9SO-0-a770454b21950985fb2741ffc3e861c9)
定义或说明一个常数组可采用如下格式:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15872.jpg?sign=1739035479-B6Acp4MFNIEOLZFGZGLIwi9VPae8GfnZ-0-92f1a318d693fc2f55ca8d756b0e2e72)
2.常对象
常对象是指对象常量,定义格式如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15873.jpg?sign=1739035479-gSVyoAxKiiIbUojxHPrIb87yBpOKyv9p-0-64805db0b952215d905424748fc0765f)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15874.jpg?sign=1739035479-ZCktCt7IC4y3rLYzXAior9p1Foc2h4sc-0-b3037da8db3260141313cf108d52bd3d)
定义常对象时,同样要进行初始化,并且该对象不能再被更新,修饰符const可以放在类名后面,也可以放在类名前面。