Effective C++在此条款中总结出两个结论

1.对于单纯常量,最好以const对象或enum替换#define

2.对于形似函数的宏,最好改用inline函数替换#define

接下来我们进行详细的探讨。

const替换#define的讨论

例如:

#define ASPECT_RATIO 1.653

原书给出的解释大意是:

你所使用的名称(ASPECT_RATIO)可能并未进入记号表(symbol table),原因也许是记号名称ASPECT_RATIO从未被编译器看见,也许在编译器开始处理源码之前他就被预处理器移走了。从而导致一个后果:当你运用此常量但获得一个编译错误信息时,这个信息也许会提到1.653而不是ASPECT_RATIO(因为所有的ASPECT_RATIO都被1.653代替),使得debug出现困难,因为你可能对1.653来自何处毫无概念,特别在多个文件都定义了ASPECT_RATIO时,无法确定是哪一个常量出了问题。

记号表:每个可重定位目标模块都有一个符号表,它包含m所定义和引用的符号的信息,在链接器的上下文中,有三种不同的符号: 1.由m定义并能被其他模块引用的全局符号 2.由其他模块定义并被模块m引用的全局符号 3.只被模块m定义和引用的本地符号
可重定位目标文件:     包含二进制代码和数据,其形式可以在编译时与其他可重定义目标文件合并起来,创建一个可执行目标文件。 

解决之道是以一个常量替换上述的宏:

const double AspectRatio = 1.653;

作为一个语言常量,AspectRatio肯定会被编译器看到,当然会进入记号表内。此外对浮点常量而言,使用常量可能比使用#define导致较小量的码,因为预处理器盲目将宏名称替换为1.653,可能导致目标码(object code)出现多份1.653,若改用常量绝不会出现相同情况。

以常量替换#define,有两种特殊情况值得注意:

1.定义常量指针。若要在头文件内定义一个指向常量的常量指针,需要些const两次

        const char* const pc = "c++";//指向常量字符串的常量char*型指针

一个小知识点是,string对象比char*类型的指针更合意,所以pc定义成下面的样式更好些

        const std::string pc("c++");//常量string对象

2.class类内的专属常量,为保证至多只有一个实体,必须声明为static。

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

对于static类型的class专属常量,若它是整数类(例如int,char,bool),只要不取他们的地址,可以声明并使用它们而无须提供定义式。

如果必须用到定义式,必须在实现文件而不是头文件如下定义

const int GamePlayer::NumTurns;//NumTurns的定义

由于NumTurns已在声明时获得初值,因此定义时不可以再设初值。如果你的编译器不支持static成员在其声明式上获得初值,可以将初值放在定义式。

static const int NumTurns;//声明 const int GamePalyer::NumTurns = 5;//定义

以enum替换#define的讨论

当你在class编译期间需要一个class常量值,如果你的编译器不允许static整数型class常量完成类内初值设定,可使用enum。

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

这样做的好处是enum不允许取地址,有时这正是我们想要的。如果不想让别人获得一个pointer或reference指向你的整数常量,enum可以帮助实现这个约束。

inline替换#define的讨论

#define CALL_WITH_MAX(a,b) f((a) > (b> ? (a) : (b))

对于这种宏的调用,有时会发生不可思议的事情

int a = 5, b = 0; CALL_WITH_MAX(++a,b);//a被累加二次 CALL_WITH_MAX(++a,b+10);//a被累加一次

这时可以使用template inline函数避免这些缺陷

template<tyoename T> inline void callWithMax(cosnt T& a,cosnt T& b) {     f(a > b ? a : b); }

《Effective C++》读书笔记 条款02 尽量以const,enum,inline替换#define的更多相关文章

  1. Effective C++学习笔记 条款02:尽量以const,enum,inline替换 #define

    尽量使用const替换 #define定义常量的原因: #define 不被视为语言的一部分 宏定义的常量,预处理器只是盲目的将宏名称替换为其的常量值,导致目标码中出现多分对应的常量,而const定义 ...

  2. 读书笔记_Effective_C++_条款二:尽量以const, enum, inline替换#define

    其实这个条款分成两部分介绍会比较好,第一部分是用const和enum替换不带参的宏,第二部分是用inline替换带参的宏. 第一部分:用const和enum替换不带参宏 宏定义#define发生在预编 ...

  3. NO.2: 尽量以const,enum,inline 替换 #define

    1.首先#define 定义不重视作用域(scope),虽然可以#undef控制,但是不美观,还存在多次替换的问题,以及没有任何封装性. 2.const XXX_XX,保证其常量性以及可控的作用域,如 ...

  4. Effective C++阅读笔记_条款2:尽量以const,enum,inline替换#define

    1.#define缺点1 #define NUM 1.2 记号NUM可能没有进入记号表,在调试或者错误信息中,无法知道1.2的含义. 改善:通过const int NUM = 1.2; 2.#dein ...

  5. Effective C++ -----条款02:尽量以const, enum, inline替换 #define

    class GamePlayer{private: static const int NumTurns = 5; int scores[NumTurns]; ...}; 万一你的编译器(错误地)不允许 ...

  6. Effective C++之条款2:尽量以const enum inline替换 #define

    本文的标题也可以改成“用编译器替换预处理器”: const double AspectRatio = 1.653; //最好使用上述代码替换下述代码: #define ASPECT_RATIO 1.6 ...

  7. 条款02:尽量以const,enum,inline替换#define

    目录 1. 总结 2. 使用const常量或enum替换宏常量 class外部的常量指针 class专属常量 1. 总结 对于单纯常量,最好以const常量或enum替换#define 对于宏代码段, ...

  8. 条款2:尽量以const, enum, inline替换#define

    原因: 1. 追踪困难,由于在编译期已经替换,在记号表中没有. 2. 由于编译期多处替换,可能导致目标代码体积稍大. 3. define没有作用域,如在类中定义一个常量不行. 做法: 可以用const ...

  9. 条款2:尽量使用const ,enum,inline替换define

    宁可使用编译器而不用预处理器 假设我们使用预处理器: #define ABC 1.56 这标识符ABC也许编译器没看到,也许它在编译器处理源码前就被预处理器移走了,于是“标识符”ABC没有进入标识符列 ...

随机推荐

  1. Oracle在线重定义(online redefinition)--将普通表改为分区表

    使用Oracle的在线重定义技术,可以将Oracle的普通表改为分区表.操作如下: STEP1:测试表是否可以在线重定义,这里以unixdev数据库的LIJIAMAN.BSTEST为例 EXEC DB ...

  2. phalcon框架与Volt 模块引擎 使用简介

      ———— 近期工作中web页面使用由C语言编写的Volt模板引擎,相比之前由js动态加载页面速度更快,更利于百度数据的抓取,现根据文档整理一下使用思路 (Volt是一个超快速和设计者友好的模板语言 ...

  3. windows用交互式命令执行python程序

    1.进入cmd命令 windows+r2.进入盘符,eg:E:3.使用dir命令查看当前文件夹下的所有目录4.使用绝对路径或者相对路径和cd命令直接进入想要到达的文件夹目录(或者使用cd命令一步一步达 ...

  4. 搜索 水题&&错误集锦

    引子: 本以为搜索的题目老师也不会检查,结果今天早上loli慢悠悠的说:“请同学们提交一下搜索的题目~”,顿时心旌摇曳,却也只能装作镇定自若的样子,点了点头.. 然后就开始了今天的疯狂做题,虽说题目都 ...

  5. (Nagios)-check_openmanage[Dell]

      Nagios->check_openmanage[Dell R7*] 2014年11月13日 下午 07:44 需求介绍: 透过Nagios监控Dell R7系列服务器硬件状态 环境信息: ...

  6. ubuntu18.04错误配置变量环境导致无法进入系统

    1.问题描述 错误配置环境变量(直接在/etc/profile文件末尾添加了export xxx),关机后一直在登录界面循环无法进入系统. ###环境变量的添加是在原有变量之后以冒号(:)分隔加入,并 ...

  7. hive新手学习随笔

    一.回顾    1.hive基于Hadoop的(存储HDFS,计算MR)    2.sql on hadoop概念            ->简化开发的操作            ->提升 ...

  8. STM32(11)——DMA

    简介: DMA:Direct Memory Access,直接存储器访问.DMA传输数据从一个地址空间复制到另外一个地址空间.当CPU初始化这个传输动作,传输动作本身就是DMA控制器来实现和完成.典型 ...

  9. 常用MySQL语法

    一.进入MySQL与退出MySQL 1.进入MySQL步骤:先打开CMD命令行:命令:C:\Users\admin> mysql -h(域名,可填或不填) -u(账号) -p(密码): 连接成功 ...

  10. transient是干嘛的

    Java的serialization提供了一种持久化对象实例的机制.当持久化对象时,可能有一个特殊的对象数据成员,我们不想用 serialization机制来保存它.为了在一个特定对象的一个域上关闭s ...