C宏——智者的利刃 愚者的恶梦!

2016-01-29 12:27 10 1 收藏

C宏——智者的利刃 愚者的恶梦!,C宏——智者的利刃,愚者的恶梦!

【 tulaoshi.com - C语言心得技巧 】

C宏——智者的利刃,愚者的恶梦!

作者:乾坤一笑

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com)

本文出自 乾坤一笑 的Blog文章

  水平不高不低的C++程序员最喜欢挂在嘴上的一句话就是:C宏,万恶之首,错误的开端,应该被废弃。
  请注意,我用了一句不敬的修饰语“水平不高不低的”。为什么这么说?因为水平低都插不上话,都在在静静地听老前辈布道呢。水平高的,比如Bane Stroustrup老人家,也只是说若干种场合下C++语言能够提供比C macro更好的解决方案,而没有完全否定C macro的价值。但是话就怕传来传去,一传就走样。久而久之,就被传成上面那句话。其实说来也很好笑:java程序员经常说java比C++好,说C++手动释放内存老搞内存泄漏;C++程序员便反驳说,那是你水平低不会用。但是谈到C宏,水平不高不低的C++程序员居然也走java的老路了——明明是自己不会用,自己知道的少,却把责任推卸到C宏上。你自己笨我管不着,但是错误的言论如果误导后人就不好了吧。:)
  本文就举几个简单的使用C宏的例子,如果这些例子用C++不用宏的语法能更好的解决,那么你一定要回复blog告诉我,这样下次我就不乱说话了。否则,笑笑很生气,后果很严重。:)

例一、用C宏,书写代码更简洁这段代码写网络程序的朋友都很眼熟,是Net/3中mbuf的实现。

struct mbuf{    struct m_hdr mhdr;    union {        struct         {            struct pkthdr MH_pkthdr; /* M_PKTHDR set */            union             {                struct m_ext MH_ext; /* M_EXT set */                char MH_databuf[MHLEN];            } MH_dat;        } MH;        char M_databuf[MLEN];        /* !M_PKTHER, !M_EXT*/    } M_dat;};
  上面的代码,假如我想访问最里层的MH_databuf,那么我必须写M_dat.MH.MH_dat.MH_databuf; 这是不是很长,很难写呀?这样的代码阅读起来也不明了。其实,对于MH_pkthdr、MH_ext、MH_databuf来说,虽然不是在一个结构层次上,但是如果我们站在mbuf之外来看,它们都是mbuf的属性,完全可以压扁到一个平面上去看。所以,源码中有这么一组宏:
#define m_next      m_hdr.mh_next#define m_len       m_hdr.mh_len#define m_data      m_hdr.mh_data... ...#define m_pkthdr    M_dat.MH.MH_pkthdr#define m_pktdat    M_dat.MH.MH_dat.MH_databuf... ...
这样写起代码来,是不是很精练呢!

例二、用C宏,实现跨平台和编译器的需要这方面的例子太好举了,一举一大摞,就从VC的库源码中随意copy一段出来吧。
#ifndef _CRTAPI1#if _MSC_VER >= 800 && _M_IX86 >= 300#define _CRTAPI1 __cdecl#else  /* _MSC_VER >= 800 && _M_IX86 >= 300 */#define _CRTAPI1#endif  /* _MSC_VER >= 800 && _M_IX86 >= 300 */#endif  /* _CRTAPI1 */#ifndef _SIZE_T_DEFINEDtypedef unsigned int size_t;#define _SIZE_T_DEFINED#endif  /* _SIZE_T_DEFINED */#ifndef _MAC#ifndef _WCHAR_T_DEFINEDtypedef unsigned short wchar_t;#define _WCHAR_T_DEFINED#endif  /* _WCHAR_T_DEFINED */#endif  /* _MAC */ #ifndef _NLSCMP_DEFINED#define _NLSCMPERROR    2147483647  /* currently == INT_MAX */#define _NLSCMP_DEFINED#endif  /* _NLSCMP_DEFINED */
请问,这些指示宏如何取代呢?如果真的是没有了这些宏,实现起来就更麻烦了吧。

例三、用C宏,自动生成代码这方面的例子也是多得很,不过有鉴于很多朋友不用很多编译器,不做嵌入式的开发,我就举个win平台的例子吧。我们知道MFC实现了windows的消息映射,比如:
ON_COMMAND(IDM_ABOUT, OnAbout)ON_COMMAND(IDM_FILENEW, OnFileNew)
它是如何实现的IDM_ABOUT和OnAbout的关联的呢?这要用到几个宏。
#define DECLARE_MESSAGE_MAP() private:  static const AFX_MSGMAP_ENTRY _messageEntries[]; protected:  static AFX_DATA const AFX_MSGMAP messageMap;  virtual const AFX_MSGMAP* GetMessageMap() const; #define BEGIN_MESSAGE_MAP(theClass, baseClass)  const AFX_MSGMAP* theClass::GetMessageMap() const   { return &theClass::messageMap; }  AFX_COMDAT AFX_DATADEF const AFX_MSGMAP theClass::messageMap =  { &baseClass::messageMap, &theClass::_messageEntries[0] };  AFX_COMDAT const AFX_MSGMAP_ENTRY theClass::_messageEntries[] =  {  #define ON_COMMAND(id, memberFxn)         { WM_COMMAND, 0, (WORD)id, (WORD)id, AfxSig_vv, (AFX_PMSG)memberFxn }, #define END_MESSAGE_MAP()   {0, 0, 0, 0, AfxSig_end, (AFX_PMSG)0 }  }; #define DECLARE_MESSAGE_MAP() private:  static const AFX_MSGMAP_ENTRY _messageEntries[]; protected:  static AFX_DATA const AFX_MSGMAP messag
                        

来源:http://www.tulaoshi.com/n/20160129/1486157.html

延伸阅读
标签: 月子
注意到这些产后尴尬了吗 我国近半数的已婚育女性有不同程度的盆底功能障碍,包括尿失禁、阴道松弛、便秘、痔疮、子宫脱垂等。许多女性错误的认为这些尴尬是生完孩子后的正常现象,对产后盆底肌肉收缩功能差等情况没有理会。生产后对盆底肌肉恢复的疏忽,容易在中老年后出现轻中度压力...
标签: 电脑入门
Visual Basic for Applications(VBA)是Visual Basic的一种宏语言,主要能用来扩展Windows的应用程式功能,特别是Microsoft Office软件,其中包括Excel、PPT、Word、Outlook等。 使用Excel的VBA开发的Excel文档,在Excel中运行时需要开启Excel的宏功能,否则此文档的VBA自动化功能将被完全屏蔽,文档的功能无法实现。 因此,在打开这些Excel...
标签: 分娩
在一夜的睡眠中,幼儿会有两个时间做恶梦:第一个是在刚睡着不到二个小时内,这时做的恶梦通常较真实,会使孩子从梦中惊醒后,觉得万分惊恐,不敢再次入睡;第二个时间是在醒前的三小时内,所做的梦往往象征着孩子对日间所发生的不愉快的事件反应。其症状在睡觉前突然醒来,且发出叫或哭泣声,脸上表情是害怕、惊吓,且全身出冷汗、呼吸急...
标签: word
Word中删除宏的技巧   在Word中,学会创建宏,不是难事,重点是要在会创建的基础之上还要会删除。当然很多的朋友,连宏都不大了解,又怎么会了解删除宏的方法呢?接下来,便是高手介绍的常用的删除单个宏的方法。 Word 打开Word 2003,单击工具菜单,指向宏,在下一级菜单再单击宏。 在宏列表框中单击要删除的宏名,...
PPT宏的应用和技巧   在PowerPoint中,我们也可以通过录制宏,来帮助我们自动完成一系列操作。 1、执行工具→宏→录制新宏命令,打开录制新宏对话框,输入一个名称,按下确定按钮进行录制。 2、根据你的需要,将要录制的过程操作一遍,然后按下停止录制按钮,退出录制状态。 3、执行工具→宏→宏命...

经验教程

412

收藏

96
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部