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

原创 2005年04月28日 11:06:00

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

作者:乾坤一笑

本文出自 乾坤一笑 的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_DEFINED
typedef unsigned int size_t;
#define _SIZE_T_DEFINED
#endif  /* _SIZE_T_DEFINED */

#ifndef _MAC
#ifndef _WCHAR_T_DEFINED
typedef 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 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 } /
 }; /
嘿嘿,就这么几个宏,就构造出一个消息数组来。

例四、用C宏,智者思维的火花说了半天了,嘴皮子都干了,举个例子大家轻松一下——看看人家老外是怎么用宏的。这个例子摘自《C专家编程》。 根据位模式构建图形图标(icon)或者图形(glyph),是一种小型的位模式映射于屏幕产生的图像。一个位代表图像上的一个像素。如果一个位被设置,那么它所代表的像素就是“亮”的。如果一个位被清除,那么它所代表的像素就是“暗”的。所以,一系列的整数值能够用于为图像编码。类似Iconedit这样的工具就是用于绘图的,他们所输出的是一个包含一系列整型数的ASCII文件,可以被一个窗口程序所包含。它所存在的问题是程序中的图标只是一串十六进制数。在C语言中,典型的16X16的黑白图形可能如下:
static unsigned short stopwatch[] = {
0x07C6,
0x1FF7,
0x383B,
0x600C,
0x600C,
0xC006,
0xC006,
0xDF06,
0xC106,
0xC106,
0x610C,
0x610C,
0x3838,
0x1FF0,
0x07C0,
0x0000
};
  正如所看到的那样,这些C语言常量并未有提供有关图形实际模样的任何线索。这里有一个惊人的#define定义的优雅集合,允许程序建立常量使它们看上去像是屏幕上的图形。
#define X )*2+1
#define _ )*2
#define s ((((((((((((((((0 /* For building glyphs 16 bits wide */
  定义了它们之后,只要画所需要的图标或者图形等,程序会自动创建它们的十六进制模式。使用这些宏定义,程序的自描述能力大大加强,上面这个例子可以转变为:
static unsigned short stopwatch[] =
{
s _ _ _ _ _ X X X X X _ _ _ X X _ ,
s _ _ _ X X X X X X X X X _ X X X ,
s _ _ X X X _ _ _ _ _ X X X _ X X ,
s _ X X _ _ _ _ _ _ _ _ _ X X _ _ ,
s _ X X _ _ _ _ _ _ _ _ _ X X _ _ ,
s X X _ _ _ _ _ _ _ _ _ _ _ X X _ ,
s X X _ _ _ _ _ _ _ _ _ _ _ X X _ ,
s X X _ X X X X X _ _ _ _ _ X X _ ,
s X X _ _ _ _ _ X _ _ _ _ _ X X _ ,
s X X _ _ _ _ _ X _ _ _ _ _ X X _ ,
s _ X X _ _ _ _ X _ _ _ _ X X _ _ ,
s _ X X _ _ _ _ X _ _ _ _ X X _ _ ,
s _ _ X X X _ _ _ _ _ X X X _ _ _ ,
s _ _ _ X X X X X X X X X _ _ _ _ ,
s _ _ _ _ _ X X X X X _ _ _ _ _ _ ,
s _ _ _ _ _ _ _ _ _ _ _ _ _ _ _ _
};
  显然,与前面的代码相比,它的意思更为明显。标准的C语言具有八进制、十进制和十六进制常量,但没有二进制常量,否则的话倒是一种更为简单的绘制图形模式的方法。
  如果抓住书的右上角,并斜这看这一页,可能会猜测这是一个用于流行窗口系统的“cursor busy”小秒表图形。我是在几年前从Usenet comp.lang.c新闻组学到这个技巧的。

  千万不要忘了在绘图结束后清除这些宏定义,否这很可能会给你后面的代码带来不可预测的后果。

  好了,今天的废话就到这里了。水能载舟,亦能覆舟,把握好手中的双刃剑,让它好好的为你服务吧,别割破了手。:)

愚者与智者的差距

愚者与智者的差距 第一章 学会尊重他人 1.智者善于倾听,愚者没有耐心 2.智者了解别人的心思,愚者表示自己的需要 3.智者记住别人的名字,愚者希望名字被记住 4.智者满脸微笑,愚者...
  • gdutcc
  • gdutcc
  • 2011年08月04日 09:10
  • 267

愚者与智者的48个差距

  • 2007年06月28日 00:09
  • 643KB
  • 下载

Java NIO——4 在多线程环境下的恶梦之终结

有人说java nio在多线程环境下编程简直就是个恶梦,其实你如果能把握住java nio API的要领,你就可以将之驾驭.   0.  一个 channal 对应一个SelectionK...

智者1号写字楼众筹:颠覆式的写字楼开发新模式来临

据悉,在写字楼存量大、去化慢的整体市场状况下,高新区某写字楼项目创下了两周认筹逾六成的销售佳绩,堪称写字楼商业房产的奇迹。记者为此专程走访了该项目,近距离地感受到了房产模式变革的魅力。 该写字楼项目...

架构师:要成为百科全书式的智者

架构师是什么?它不是行政职务,也不简单的一种职称。对于一线程序员来说,拥有方方面面各种知识的架构师,是一位百科全书式的智者。 AD:2014WOT全球软件技术峰会北京站 课程视频发布 【51C...

愚者指名自己的辩护人_纪中2046_最短路_维包一生推,gosick大法好!

输入第一行两个整数 N,M,表示点数和边数。 接下来一行 N 个整数,第 i 个正整数表示 Pi。 接下来 M 行,每行两个整数 u,v,表示有一条无向边连接了 u 和 v。输出输出 N 行,每行...
  • jpwang8
  • jpwang8
  • 2016年09月13日 16:28
  • 153

Java Lombok 减少代码冗余 利刃

前言: 现在是ORM框架大行其道的时代,有着如Hibernate这样重量级的框架,提供Java数据的双向映射,也有如Mybatis这样的轻量级框架,提供Java 无论使用的是哪种流行的框架,在其中都少...

2046. 愚者指名自己的辩护人

2046. 愚者指名自己的辩护人  输入 第一行两个整数 N,M,表示点数和边数。 接下来一行 N 个整数,第 i 个正整数表示 Pi。 接下来 M 行,每行两个整数 u,v,表示有一条无向边连接了 ...

恶梦射手学习笔记

最近学习完了恶梦射手游戏的制作,现在将其中一些需要记下的点记下来。 本人使用的是Uinty5.0,视频教程是Unity4.5。 游戏跟着做完后有着一些修改,发布到Android手机上运行正常,操作是...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:C宏——智者的利刃,愚者的恶梦!
举报原因:
原因补充:

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