浅析C++里面的宏

原创 2003年06月30日 13:45:00

说到宏,恐怕大家都能说出点东西来:一种预处理,没有分号(真的吗?)。然后呢?
嗯.......茫然中......
好吧,我们就从这开始说起。
最常见的宏恐怕是#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。  

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

说到宏,恐怕大家都能说出点东西来:一种预处理,没有分号(真的吗?)。然后呢?嗯.......茫然中......好吧,我们就从这开始说起。最常见的宏恐怕是#include了,其次就是#define还有....
  • jiangredsheep
  • jiangredsheep
  • 2005年06月30日 16:47
  • 903

C语言字符串中使用宏替换

1 背景     在定义一个字符串时,希望有一部分内容可以使用宏替换。当尝试使用如下方式来实现时,发现并不能达到目的。#define XXX programmer char str[] = “I l...
  • Q1302182594
  • Q1302182594
  • 2015年07月21日 11:04
  • 2635

C/C++语言中头文件的保护宏

保护宏就是C/C++头文件开始处的宏判断和宏定义,可以避免该头文件被多次加载执行而导致编译错误的宏。...
  • zzfenglin
  • zzfenglin
  • 2016年04月09日 11:36
  • 1756

[c、c++]宏中"#"和"##"的用法(zz)

[c、c++]宏中"#"和"##"的用法(zz)宏中"#"和"##"的用法一、一般用法我们使用#把宏参数变为一个字符串,用##把两个宏参数贴合在一起.用法:#include#include...
  • blackbillow
  • blackbillow
  • 2009年01月22日 16:51
  • 36994

C++/C 宏定义(define)中#和## 的含义

#是字符串化的意思,出现在宏定义中的#是把跟在后面的参数转成一个字符串; ##是连接符号,把参数连接在一起。 例子: #include using namespace std; #define ...
  • fireblue1990
  • fireblue1990
  • 2017年02月17日 14:57
  • 340

Android C++层如何判断ProjectConfig.mk中定义的宏

A  16:15:56 C++层如何判断ProjectConfig_PD1218MA.mk中定义的宏,哪位大神知道 B 16:20:03 在Android.mk里面判断 #ifdef __xx...
  • syh63053767
  • syh63053767
  • 2012年11月03日 14:31
  • 846

宏里面的作用域

//----------------------加上一对花括号是为了在同一个函数里可以使用多个带定义的MYLOG,不同作用域了 #define LOG 0 #ifdef LOG  #includ...
  • h1023417614
  • h1023417614
  • 2015年03月26日 17:28
  • 845

深入了解View(二)—— layout布局流程分析

layout过程简介measure过程结束后,视图的大小就已经测量好了,接下来就是layout的过程了。正如其名字所描述的一样,这个方法是用于给视图进行布局的,也就是确定视图的位置。ViewRoot的...
  • wuseyukui
  • wuseyukui
  • 2015年07月25日 11:16
  • 652

C++函数前加宏表示的意思

在VC编程中,也许大家会遇到如下函数定义:LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)...
  • smallmuou
  • smallmuou
  • 2009年10月21日 11:13
  • 2101

C++11读书笔记—1(长期历史遗留问题,宏定义与类型)

一、宏定义的补全 1.C宏的补全 C的精髓宏定义更有更好的兼容请看下面代码,VS2015在部分没有做提示,但是可以用。但在标准C++0x版本不能用, #include #define LOG(....
  • dajiabudongdao
  • dajiabudongdao
  • 2016年07月28日 18:59
  • 590
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:浅析C++里面的宏
举报原因:
原因补充:

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