【Effective C++读书笔记】条款02 尽量以const, enum, inline 替换 #define

条款02列了三种情况说明为什么要尽量以const,enum,inline替换#define。下面一一介绍。

1. const替换#define

#define ASPECT_RATIO 1.653

当你使用#define时,在编译器进行编译之前,预处理器会将所有的ASPECT_RATIO替换为1.653。所以我们可能会遇到这种情况,就是当出现一个编译错误时,错误信息会提到1.653,而不是ASPECT_RATIO,因为在编译之前ASPECT_RATIO已被替换。当程序很大时,我们对1.653来自何方毫无概念,对于追踪它会浪费很多时间。
解决的方法是以一个常量替换上述的宏(#define):

const double AspectRatio = 1.653;

书中下面这段话我没太明白啥意思,看懂的可以留言互相讨论。

对浮点常量(floating point constant, 就像本例)而言,使用常量可能比使用#define导致较小量的代码,因为预处理器“盲目地将宏名称ASPECT_RATIO替换胃1.653”可能导致目标码(object code)出现多份1.653,若改用常量AspectRatio绝不会出现相同情况。

使用const有两点需要注意的地方:

1)常量指针

若要在头文件中定义一个常量指针,必须写两次const。

const char* const authorName = "Scott Meyers";
2)class专属常量
class GamePlayer {
private:
  static const int NumTurns = 5;  // 常量声明式
  int scores[NumTurns];
}

static保证了常量至多只有一个实体。但是,我们在此处看到的NumTurns只是一个声明式。
通常情况下,C++要求我们对使用的任何东西都提供一个定义式,但如果它是个专属常量又是static且为整数类型(本例),则需特殊处理。如果我们是使用时不取常量的地址,我们可以声明并使用而无须提供定义式。否则就必须提供如下定义式:

const int GamePlayer::NumTurns;  

上式需要放进一个实现文件而非头文件。由于class常量已在声明时获得初值,因此定义时不可再设置初值。

2. 使用enum替换#define

刚才的例子:

class GamePlayer {
private:
  static const int NumTurns = 5;  // 常量声明式
  int scores[NumTurns];
}

有时候,有的编译器不允许static成员在其声明式上获得初值。所以只能将初值放在定义式:

// 头文件
class GamePlayer {
private:
  static const int NumTurns = 5;  // 常量声明式
};
//实现文件
const int GamePlayer::NumTurns = 5;

但是,类中还有一个变量int scores[NumTurns],该变量必须在指定NumTurns的值(编译器要求),这就出现矛盾了。如何解决这个问题呢,enum hack就出场了!

class GamePlayer{
private:
  enum {NumTurns = 5};
  int scores[NumTurns];
}

我们让 NumTurns成为5的一个记号名称,这样就没问题了。

enum hack的行为某方面说比较像#define而不像const,例如取一个const的地址是合法的,但取一个enum的地址就不合法,而取一个#define的地址也不合法。

3. inline替换#define

有时候我们会用#define定义一个函数:

#define GET_THE_MAX(a, b) a > b ? a : b

这种写法有时会发生不可思议的事情:

int a = 10, b = 0;
GET_THE_MAX(++a, b);            // a被累加两次
GET_THE_MAX(++a, b+20);         // a被累加一次

a的递增次数竟然取决于“它被拿来和谁作比较”!
#define定义的宏函数不会招致函数调用带来的额外开销,但是出现这种情况我们也是不希望的。
此时,我们可以使用inline函数完美解决这个问题。

inline int getTheMax(int a, int b){
    return a > b ? a : b;
}

如果类型不固定为int,可以使用template:

template <typename T> inline T getTheMax(const T&a, const T&b){
    return a > b ? a : b;
}

当我们在调用时,

int a = 10, b = 0;
getTheMax(++a, b);            // a被累加一次
getTheMax(++a, b+20);         // a被累加一次

不论与谁作比较,a只会累加一次。

4.结语

有了consts, enums和inlines,我们对预处理器(特别是#define)的需求降低了,但并非完全消除。#define仍然是必需品,而#ifdef/#ifndef也继续扮演控制编译的重要角色。
对于#define的使用,需要记住的两点:

  1. 对于单纯变量,最好以const对象或enum替换#define。
  2. 对于形似函数的宏(macros),最好改用inline函数替换#define。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容