![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=1739690212-2uAfN8VSqPpmwTZLcBmrd6R7hlJAYfHt-0-0ea2492ace3ae0c1b049704064536329)
预处理器无论在什么时候遇到了这样的指令,任何出现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=1739690212-xS2FwkiLxHXv3uw2Dr9LIteAkvrEOcBk-0-456b3d085599d67e95f5ca7b5d62bd09)
【代码详解】
在该例中,使用宏预处理器定义了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=1739690212-yZ7lRbPTs27pB9Gv5FRb7J60I7uhiuXb-0-10aa694af9746f4b65242cb9977b4f8f)
图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=1739690212-0XKFATSRWT1D0BFDejxkIr6hStdrg1N0-0-325c7061a2f68b8010632d62be0f9c17)
像122这样的数字在程序中被称为魔法数字。一个魔法数字是hard-coded数字,它在代码中没有任何意义—122表示什么呢?是转换率还是其他什么呢?它是不明确的。在一些复杂的程序里,通常很难判断一个hard-coded数字具体代表什么。
下面的小段代码是清晰的:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15768.jpg?sign=1739690212-HSfLUq9IXgixWSluMFMJ0LYELr0iv7Xf-0-47ad2dd9099b37573c11616f95404f45)
其次,#defined数字可以使得程序更加容易被修改。假设将转换率从122变成123,程序需要进行相应的调整。考虑下面的代码:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15769.jpg?sign=1739690212-m6Ku5bfKKHSbIicXqHAVwQ8WZ5OGshTZ-0-5bf33844682e8794141ee40aac7a2ade)
为了改变成新的转换率,必须将前面4个语句中的数字改变。但是第5个语句呢?这里的122是不是和其他的122具有相同意义呢?如果是,它就应该被改变;如果不是,就不需要改变,或者也许在其他地方中断。
现在考虑使用了#defined的情况,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15770.jpg?sign=1739690212-EPnSVfSwHjivCkoovtFDigUejjqFQMb2-0-14aefd73675783eec74c80c747a5f7c6)
这时改变转换率只要改变一个数字,代码如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P122_15771.jpg?sign=1739690212-8UtnbAr2N0OTkswxaMGQA3A8IshH5qYI-0-f9bcda0d0bb20d06e8570ee42512853e)
现在正确改变了转换率,并且不用担心将每页的行数改变。
6.5.3 const修饰符
常类型是指使用类型修饰符const说明的类型。常类型的变量或对象的值不能被更新。
提示
编译器通常不为普通const常量分配存储空间,而是将它们保存在符号表中,这使得它成为一个编译期间的常量,没有了存储与读内存的操作,使得它的效率也很高。因此,定义或说明常类型时必须进行初始化。
1.一般常量
一般常量是指简单类型的常量。这种常量在定义时,修饰符const可以用在类型说明符前,也可以用在类型说明符后,例如:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15870.jpg?sign=1739690212-oxkmMhRICkVfofQORtWGXDQmK4r1Oyfh-0-b011a9298832fe5729763c897219cede)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15871.jpg?sign=1739690212-nk6yA5uP0ikUQymoKX1tBxViUm0yPnJP-0-d6890927c59159322d7ece09e09c1952)
定义或说明一个常数组可采用如下格式:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15872.jpg?sign=1739690212-UOcManhUzkOlBk3P3gHhzQBEj0fIowe6-0-1247c1c88ccbc79b656685d3894db815)
2.常对象
常对象是指对象常量,定义格式如下:
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15873.jpg?sign=1739690212-tirZ0KQbPPr30BMJKYKDCwtFMnj7dalD-0-6392fff356fc7b28bc74c204023d3f3b)
或
![](https://epubservercos.yuewen.com/A8BCBE/16499866604816706/epubprivate/OEBPS/Images/Figure-P123_15874.jpg?sign=1739690212-x2Le9qbgC62gKkxC9D9P7l6Nahn4LASa-0-43259c05c418c088dcfee777d305c1e8)
定义常对象时,同样要进行初始化,并且该对象不能再被更新,修饰符const可以放在类名后面,也可以放在类名前面。