浅析C++里面的宏。。。[ZZ]

来源:互联网 发布:javascript 获取前url 编辑:程序博客网 时间:2024/05/16 01:10





说到宏,恐怕大家都能说出点东西来:一种预处理,没有分号(真的吗?)。然后呢?

嗯.......茫然中......

好吧,我们就从这开始说起。

最常见的宏恐怕是#include了,其次就是#define还有.......

还是从宏的用途分类吧:

1、#include 主要用于包含引用文件,至今其地位无人能替代;

2、注释掉代码。例如:

#if 0

.......

#endif;

这种机制是目前注释掉代码的最佳选择,为摩托罗拉公司员工所普遍采用;

3、代码版本管理。例如:

#ifdef DEBUG

file://调试版本

#else

file://非调试版本

#endif;

4、声明宏。例如:

#define DECLARE_MESSAGE(x) x();~x() file://有没有分号?哈哈

//........

class A

{

public:

DECLARE_MESSAGE(A);

..............

}

想起什么了,呵呵:)对,VC里面有好多这样的东东,有空我会写《我的VC历程》,到

时候会把VC里的各种宏详细的解释一下,那可是一个庞大的工程:)

5、符号常量。例如:

#define PI 3.14159

6、内联函数。例如:

#define CLEAR(x) ((x)=0)

7、泛型函数。例如:

#define ABS(x) ((x)>0? (x):-(x))

x=3 没问题! x=1.3 也没问题!

如果是这样呢:

#include <iostream.h>

#define A(x) ((x)>0? (x):-(x))

void main()

{

int i=-1;

cout<<A(1)<<endl;

cout<<A(++i)<<endl;

}

有问题了,不过以后再说,大概讲const or inline 时会说的:)

8、泛型类型。例如:

#define Stack(T) Stack__ ##T

#define Stackdeclare(T) class Stack(T) {.....}

Stackdeclare(int);

Stackdeclare(char);

.......

Stack(int) s1;

Stack(char) s2;

9、语法扩展。例如:

Set<int> s;//假设Set为一个描述集合的类

int i;

FORALL(i,s);

.......

宏最大的问题便是易引起冲突,例如:

libA.h:

#define MACRO stuff

同时:

libB.h:

#define MACRO stuff

下面我们对他们进行引用:

user.cpp:

#include "libA.h"

#include "libB.h"

.............

糟糕,出现了重定义!

还有一种冲突的可能:

libB.h:(没有定义宏MACRO)

class x { void MACRO(); ...........};

那么程序运行期间,libA.h中的宏讲会改变libB.h中的成员函数的名字,导致不可预料

的结果。

宏的另一个问题,便是如7中出现的问题,如果你把7中的x设为'a',程序也不会给出任

何警告,所以他是不安全的。

针对以上的问题,我们说:

1、尽可能的少用公用宏,能替换掉就替换掉

2、对那些不能替换的宏,使用命名约定

1、符号常量预处理程序我们可以用const or enum 来代替:

const int TABLESIZE=1024;

enum { TABLESIZE=1024 };

2、非泛型内联函数的预处理程序可以使用真正的内联函数来代替:

inline void clear(int& x) {x=0;}

奥,对了,还有这样一种情况:

#define CONTROL(c) ((c)-64)

..........

switch(c)

{

case CONTROL('a') : ......

case CONTROL('b') : ......

case CONTROL('c') : ......

case CONTROL('d') : ......

..........

}

这时候就不能单独使用内联函数来取代了,因为case标签禁止函数调用,我们只好

做如下转换:

inline char control(char c) { return c+64; }

...........

switch(control(c))

{

case 'a':.....

case 'b':.....

case 'c':.....

case 'd':.....

........

}

当然这样做是以牺牲时间作为代价的(你想想为什么:))

3、对于泛型预处理程序,我们可以用函数模板或类默板来代替:

template<class T>

T ABS(const T& t) { return t>0 ? t : -t; }

template<class T>

Class Stack { ............ };

4、最后对于语法扩展程序几乎都可以用一个或多个C++类代替:

Set<int> s;

int i;

Set_iter<int> iter(s);

while(iter.next(i))

...........

与使用宏相比,我们只是牺牲了一点程序的简洁性而已。

当然并不是所有的宏都能替换(我们也并不主张替换掉所有的宏!),对于不能替换的

宏,我们应该对他们实行命名约定,例如:

#define COMPANY_XYZ_LIBABC_MACRO stuff

同时我们也要采取一定的方法,进行预防:

#ifndef COMPANY_XYZ_LIBABC_MACRO

#define COMPANY_XYZ_LIBABC_MACRO stuff

#endif

当然,在程序库实现内部定义的宏没有这个约束:

my.cpp:

#define MACRO stuff

........

我们给出几个常见的宏:

#define A(x) T_##x

#define Bx) #@x

#define Cx) #x

我们假设:x=1,则有:

A(1)=======T_1

B(1)======'1'

C(1)======"1"

还有一个比较常见的宏:_T

TCHAR tStr[] =
_T("t code");

_T宏的作用就是转换成TCHAR

原创粉丝点击