宁可使用编译器而不用预处理器

  假设我们使用预处理器:

#define ABC 1.56

  这标识符ABC也许编译器没看到,也许它在编译器处理源码前就被预处理器移走了,于是“标识符”ABC没有进入标识符列表(symbol table)中。但是当我们编译程序遇到个错误信息时,可能会带来困惑,因为这个错误信息可能会提到1.56而不是ABC,而下面例子在vs2015上编译的时候,错误信息提示ABC未定义的标识符。假如这个ABC不在我们自己定义的头文件中,我们根本无法知道其来源,追踪不到它。
解决之道就是使用一个常量代替宏定义:

const double abc=1.56;

  这样的话abc就是一个标识符,肯定会被编译器看到,必然会进入标识符列表。

 #include<iostream>
using namespace std; #define ABC 3
const int a = ; int func(const int* n)
{
return *(n + );
} int main(void)
{
int x,y;
x = func(&ABC);//vs2015会提示表达式必须为左值或者标识符
cout << x << endl; y = func(&a);
cout << y << endl;
return ;
}

常量定义特殊情况说明:
  (1).由于常量定义被放在头文件中,指针声明为const,假如是char*字符串的话,const就要写两次。

const char* const str="Burgess";

  string对象往往写成:

const std::string("Burgess");

  (2).class专属常量。为了将作用域限定在class内,必须使它成为一个成员变量。为了保证这个常量只有一个实体,要将它声明为static。

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

  一般情况下,我们还需要在class外进行定义它。

const int A::Num;

  由于已经在类里面设了初值,这里不必再初始化。(旧编译器不支持在类里面为static const常量设初值),那么就要:

 class A
{
private:
static const int Num;//声明式
};
const int A::Num=;//定义式

这个定义式放在实现文件而不是头文件中。
需要说明的是,类里面的函数使用这个变量时,也需要声明为static。
假如上面的数组大小坚持使用一个标识符来代替怎么办呢,因为编译器必须在编译期间知道数组大小,那么我们就可以使用一个枚举类型的数值代替int型数值。

 class A
{
private:
enum{Num=};
int scores[Num];
};

这里enum类似#define,不能取地址,只需要获取其值。假如我们不想让别人获取指向某个常量的指针或者引用,可以使用enum。
让我们再返回宏定义。举个误用宏定义的例子。有时我们使用#define实现类似函数的宏定义,但是没有函数调用引来的额外开销。看下面:

 #define CALL_MAX(a,b) ((a)>(b))?(a):(b)
int main(void)
{
int ret1,ret2,a = , b = ;
ret1 = CALL_MAX(++a, b);//现在a=7,不可思议
cout<< ret1<<endl; ret2 = CALL_MAX(++a, b+);//a=6
cout << ret2 << endl;
return ;
}

为什么 CALL_MAX(++a, b+10),此时a=6?
把参数代入表达式就可以知道:

((++a)>(b+))?(++a):(b+)

可以看成6>10吗,显然不是。那么就返回b+10=10,而不再执行++a,故a也不会再加1了。

如果不想要这种未知行为的#define,但是还想要宏定义的效率、安全性和可预知性(一切掌握在我们自己手里),那么inline模板函数是个好选择:

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

  这样的话,不会出现上面#define出现的++a加1进行了两次的情况,此时max1是真正的函数。

请记住:
  (1).单纯的常量,最好使用const或者enum代替#define;
  (2).类似函数的宏,最好以inline函数代替#define。

条款2:尽量使用const ,enum,inline替换define的更多相关文章

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

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

  9. Book. Effective C++ item2-尽量使用const, enum, inline替换#define

    ##常规变量 c++里面的#define后面的定义部分,是不算代码的一部分的.所以如果你使用#define: #define ASPECT_RATIO 1.653 你希望这个代号ASPECT RATI ...

随机推荐

  1. vue大文件分片上传插件

    最近遇见一个需要上传百兆大文件的需求,调研了七牛和腾讯云的切片分段上传功能,因此在此整理前端大文件上传相关功能的实现. 在某些业务中,大文件上传是一个比较重要的交互场景,如上传入库比较大的Excel表 ...

  2. redis学习 --Hash

    一:我们可以将Redis中的Hash类型看成具有String Key和String Value的map容器.所以该类型非常适合于存储值对象的信息.如Username.Password和Age等.如果H ...

  3. MyCat的启动

    启动MyCat: ./mycat start 查看启动状态: ./mycat status 停止: ./mycat stop 重启: ./mycat restart 

  4. POJ 1502 MPI MaeIstrom ( 裸最短路 || atoi系统函数 )

    题意 : 给出 N 个点,各个点之间的路径长度用给出的下三角矩阵表示,上上角矩阵和下三角矩阵是一样的,主对角线的元素都是 0 代表自己到达自己不用花费,现在问你从 1 到 N 的最短路,矩阵的 x 代 ...

  5. [CF846A]Curriculum Vitae题解

    枚举一个点,假设它一定符合条件,据此珂以\(O(n)\)算出要删去几个点 于是就\(O(n^2)\)解决了,貌似加一个前缀和可以在\(O(n)\)的时间复杂度内解决本问题,但对于这个数据范围来说\(O ...

  6. 【转】Office 2003 EXCEL多窗口打开

    转自:http://blog.csdn.net/god_is_gril/article/details/8992587 1.注册表备份开始/运行,输入regedit回车,打开注册表.在注册表界面定位到 ...

  7. 2019 上海网络赛 J stone name (01背包)

    题目:https://nanti.jisuanke.com/t/41420 题意:给你一个集合,然后让你拆成两个集合 x,y    求满足  x>y  &&  x-(x集合中最小 ...

  8. ReactNative的学习笔记

    一.安装nodejs 查看是否安装:npm -v 二.安装react-native命令工具 npm install -g react-native-cli 三.查看 react-native --he ...

  9. HDU 5179 beautiful number (数位dp / 暴力打表 / dfs)

    beautiful number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  10. video.js播放rtmp

    项目中要用到rtmp直播和点播.要求:点播能够调整播放进度 开始用腾讯提供的播放器,老卡,画质差,很多时候播不出来,rtmp点播还不能快进. 后来用Wowza自带的flash rtmp播放器,有源码 ...