【Effective C++】条款2:尽量以const,enum,inline替换#define

一.const替换#define

#define的两个缺点:

  1. 不便于调试,编译器看到的是宏替换后的值,报错时显示也是替换后的内容,不便于定位错误
  2. 目标代码中出现多个替换后的内容。因为宏替换在预处理阶段完成,宏是不会进入记号表的。

替换方案:使用const

cpp 复制代码
//#define ARPECT_RATIO 1.653;
const double ArpectRatio = 1.653;

用#define替换常量时的两种特殊情况:

1.定义常量指针

cpp 复制代码
//#define NAME "Effective C++"
const char* const Name = "Effective C++";

2.定义特定于类的常量

将常量作用域限制在类中,并确保只有一个常量的副本,就必须设置为类的静态成员

cpp 复制代码
class GamePlayer
{
	//#define NUMTURNS 5;//这是无效的,#define只是无脑替换,无法将将作用域限制在类的内部
	static const int NumTurns = 5;//静态成员变量声明
	int scores[NumTurns];//使用静态成员变量
};

通常情况下,C++要求为所使用的任何东西提供定义,但类的静态整数类型常量(如bool,int,char)是个特例,可以声明时就给初值,在不取它们地址地址的情况下,可以直接使用,不用提供定义。

如果你在声明时给了初值,定义时就不能给值了。如果声明时没有给初值,定义时就能给值了。

cpp 复制代码
class GamePlayer
{
	//#define NUMTURNS 5;//这是无效的,#define只是无脑替换,无法将将作用域限制在类的内部
	static const int NumTurns = 5;//静态成员变量声明
	//使用静态成员变量
};

//const int GamePlayer::NumTurns = 6;//不允许的
const int GamePlayer::NumTurns;

二.enum替代#define

枚举类型的值可以在需要int的地方使用

cpp 复制代码
class GamePlayer
{
	enum{NumTurns = 5};
	int scores[NumTurns];//使用枚举常量,枚举更像#define,不能取地址
};

三.inline替代#define

我们实现如下宏函数:

cpp 复制代码
#define CALL_WITH_MAX(a, b) f((a) > (b) ? (a) : (b)) //f是一个函数

尽管我们已经很小心了,为每个参数都加上了括号,但问题防不胜防:

cpp 复制代码
int a = 5, b = 0;
CALL_WITH_MAX(++a, b);
CALL_WITH_MAX(++a, b + 10);

由于#define只是无脑替换,根据执行逻辑,第一个表达式a会被++两次,第二个a会++一次。这绝不是我们想要的结果,我们期望a无论如何只会++一次!!!

替代方案:模版+内联函数,既能获得宏的效率,又能保证函数行为可预测与类型安全

cpp 复制代码
template<class T>
inline void CallWithMax(const T& a, const T& b)
{
	f(a > b ? a : b);
}
相关推荐
观音山保我别报错16 分钟前
列表,元组,字典
开发语言·python
**蓝桉**27 分钟前
数组的执行原理,java程序的执行原理
java·开发语言
waeng_luo39 分钟前
[鸿蒙2025领航者闯关] 表单验证与用户输入处理最佳实践
开发语言·前端·鸿蒙·鸿蒙2025领航者闯关·鸿蒙6实战·开发者年度总结
高频交易dragon41 分钟前
5分钟和30分钟联立进行缠论信号分析
开发语言·python
ULTRA??43 分钟前
C/C++函数指针
c语言·开发语言·c++
还没想好取啥名44 分钟前
C++11新特性(一)——自动类型推导
开发语言·c++·stl
我是华为OD~HR~栗栗呀1 小时前
华为OD-C面经-23届学院哦
java·c++·python·华为od·华为·面试
xiaozi41201 小时前
Ruey S. Tsay《时间序列分析》Python实现笔记:综合与应用
开发语言·笔记·python·机器学习
西贝爱学习1 小时前
Visual Studio下载地址,vs2022安装程序
c++
天赐学c语言1 小时前
12.5 - 二叉树的最近公共祖先 && 构造函数和析构函数可以是虚函数吗
c++·二叉树·虚函数