【C++】灵活使用C/C++宏定义

原创 2016年05月31日 20:36:45

灵活使用C/C++宏定义

 #define宏定义在C语言中有广泛使用。对于C++语言,《Effective C++》第二章告诉我们,在C++中尽量用const、enum、inline替换#define,可以避免很多陷阱。其中const用于定义常量,enum用于定义枚举,inline用于定义内联函数。
 尽管《Effective C++》第二章指出#define宏定义的诸多陷阱,这不代表作者不希望我们使用#define。#define的作用是,在预编译阶段,编译器将#define定义的代码段替换到使用相应宏定义的位置上去,相当于直接将代码替换宏,如果宏定义是安全的、简单的,那么#define宏定义会有很大的作用。很多著名C++第三方库使用了大量的宏定义,可以说明这一点。
 来看看Eigen矩阵库。Eigen是基于模板的C++开源矩阵库,使用或者支持Eigen的知名项目有Google的TensorFlow、Point Cloud Library、OpenCV等。Eigen运用了大量的模板,为了管理这些模板,Eigen使用了大量的宏,我摘要一段如下,其中##符号表示连接两个字符串:

#define EIGEN_MAKE_TYPEDEFS(Type, TypeSuffix, Size, SizeSuffix)   \
typedef Matrix<Type, Size, Size> Matrix##SizeSuffix##TypeSuffix;  \
...
#define EIGEN_MAKE_TYPEDEFS_ALL_SIZES(Type, TypeSuffix) \
EIGEN_MAKE_TYPEDEFS(Type, TypeSuffix, 2, 2) \
EIGEN_MAKE_TYPEDEFS(Type, TypeSuffix, 3, 3) \
...
EIGEN_MAKE_TYPEDEFS_ALL_SIZES(int,                  i)
EIGEN_MAKE_TYPEDEFS_ALL_SIZES(float,                f)
...

上述写出的宏定义代码,定义了

Matrix2i<->Matrix<int, 2, 2>;
Matrix2f<->Matrix<float, 2, 2>;
Matrix3i<->Matrix<int, 3, 3>;
Matrix3f<->Matrix<float, 3, 3>

四个数据类型。用上述宏定义的方法可以用很少的代码,扩展定义更多的类型。
 再来看看VTK。我展示VTK用于定义set/get函数的代码:

//宏定义
#define vtkSetMacro(name,type) \
virtual void Set##name (type _arg) \
  { \
  vtkDebugMacro(<< this->GetClassName() << " (" << this << "): setting " #name " to " << _arg); \
  if (this->name != _arg) \
    { \
    this->name = _arg; \
    this->Modified(); \
    } \
  } 
#define vtkGetMacro(name,type) \
virtual type Get##name () { \
  vtkDebugMacro(<< this->GetClassName() << " (" << this << "): returning " << #name " of " << this->name ); \
  return this->name; \
  } 
  //定义set/get函数
  vtkSetMacro(CurrentCursor,int);
  vtkGetMacro(CurrentCursor,int);

上述代码很方便地定义了
virtual void SetCurrentCursor(int _arg);
virtual int GetCurrentCursor();

这就免得我再去手写一大堆代码。可见宏定义用于定义是很方便的。
另外我附上如果根据操作系统宏来定义不同数据的示例代码:
#ifdef _WIN32
#define SMARTPOINTER(type) std::tr1::shared_ptr<type> SP;
#else
#define SMARTPOINTER(type) std::shared_ptr<type> SP;
#endif
//下面的代码编译错误
//#define SMARTPOINTER(type) \
//#ifdef _WIN32 #define std::tr1::shared_ptr<type> SPA; \
//#else #define std::shared_ptr<type> SPA; #endif

这里看出,如果#define内部还有复杂的宏命令,则编译器很难给出正确的编译结果。尤其是使用旧版本编译器,必须要注意宏定义不能过于复杂。

版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

C/C++宏连接符'##'和'#'的用法

C(和C++)中的宏(Macro)属于编译器预处理的范畴,属于编译期概念(而非运行期概念)。下面对常遇到的宏的使用问题做了简单总结。 关 于#和## 在C语言的宏中,#的功能是将其后面的宏...

C++/C宏定义中## 连接符与# 符的含义

#是字符串化的意思,出现在宏定义中的#是把跟在后面的参数转成一个字符串 ##是连接符号,把参数连接在一起 #include using namespace std; #define ...
  • boywgw
  • boywgw
  • 2016年10月10日 10:53
  • 482

C++宏的使用(一)

 C/C++宏的使用1. 防止多重包含 22. 条件编译 23. 定义字面值常量 24. 定义为函数 25. 可变参数宏 36. 宏组合 36.1 一般用法 46.2 当宏参数是另一个宏的时候 46....

C++ 语言宏定义函数的使用(定义单行和多行)

昨天笔试问的,不会….. 在软件开发过程中,经常有一些常用或者通用的功能或者代码段,这些功能既可以写成函数,也可以封装成为宏定义。那么究竟是用函数好,还是宏定义好?这就要求我们对二者进行合理的取舍。宏...

C/C++ 宏定义中#、##、#@的区别

#表示:对应变量字符串化   ##表示:把宏参数名与宏定义代码序列中的标识符连接在一起,形成一个新的标识符 连接符#@:它将单字符标记符变换为单字符,即加单引号。例如: #define B(x)...

C/C++宏定义中#与##区别

// #表示:对应变量字符串化 // ##表示:把宏参数名与宏定义代码序列中的标识符连接在一起,形成一个新的标识符 #include #define trace(x, format) prin...
  • huang_xw
  • huang_xw
  • 2012年06月09日 15:00
  • 11518

构建c/c++项目时添加宏定义

有时候我们希望在构建时能够在命令行添加一些宏定义,改变程序行为。一个典型应用示例是代码里通过检查是否定义了 DEBUG 宏,来决定是否输出调试信息。编译器一般提供命令行选项支持这种做法,例如使用 ...

C++ 宏定义 详解

一、#define的基本用法     #define是C语言中提供的宏定义命令,其主要目的是为程序员在编程时提供一定的方便,并能在一定程度上提高程序的运行效率,但学生在学习时往往不能 理解该命令的本...

C++中如何更好使用宏定义#define

首先我们需要明确一点,在c++中不提倡使用宏。我们可以使用const代替,来定义一般的常量。对于一些函数,可以考虑inline关键字,可以达到同样的高效率。 1.为什么c++中不提倡使用宏呢?  ...
  • zhu2695
  • zhu2695
  • 2013年06月22日 21:12
  • 2279

C++宏定义详解

原地址:http://blog.chinaunix.net/uid-21372424-id-119797.html 一、#define的基本用法     #define是C语言中提供的...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:【C++】灵活使用C/C++宏定义
举报原因:
原因补充:

(最多只允许输入30个字)