X Macro
30年前我念大学时从一个朋友那里学来的一个技巧。
它是汇编语言的一个宏,但很容易转换为C语言宏。
我一直在使用它,但有意思的是我还从没在别人的代码中看到过。现在该我把这个小技巧传递下去了。
让我们举个陈腐的栗子。假设我们有一个头文件叫color.h,里面有一个颜色的宏:
enum Color { Cred, Cblue, Cgreen };
在相应的源文件color.c中,为了正确的打印颜色,有一个字符串数组:
static char *ColorStrings[] = {"red", "blue", "green"};
我们可以这样使用:
enum Color c;
...
printf("the color is %s\n",
ColorStrings[c]);
到目前为止一切都很好。随着时间推移,假如新加入一个颜色:
enum Color{ Cred, Cyellow, Cblue, Cgreen };
是的,假如我们忘记更新数组ColorStrings[]了,打印Cyellow却输出了“blue”,更糟糕的是,如果打印Cgreen会造成数组越界。
(作为一个聪明的程序员,你是不可能犯这样的错误的,对么?)
主要问题是在enum和数组之间没有语义连接。 通常的解决办法是添加一个单元测试包。
但如果我们能找到一个连接enum和数组的方法,从而在编译时检测到此类错误,岂不美哉?
--- X 宏 ---
这是它的功能么?
它能做到这一点么?
X宏如下:
#define COLORS \
X(Cred, "red") \
X(Cblue, "blue") \
X(Cgreen, "green")
把这个放在color.h中。接下来的是颜色枚举的定义:
#define X(a, b) a,
enum Color { COLORS };
#undef X
在源代码文件color.c中这样定义数组:
#define X(a, b) b,
static char *ColorStrings[] = { COLORS };
#undef X
可以看出,我们重新定义了X宏,以便提取出必要的信息而忽略其它。
正确的宏管理在这里得以体现,因为如果X已经定义过#define X将会抱怨,而#undef保证了这一点不会发生。
现在如果再添加一个颜色将变得非常简单:
#define COLORS \
X(Cred, "red") \
X(Cyellow, "yellow") \
X(Cblue, "blue") \
X(Cgreen, "green")
enum和数组都自动得到了更新,看起来很美妙是不是。有经验的程序员会立刻明白可以有更复杂的设计:
#define COLORS \
X(red) \
X(blue) \
X(green) #define X(a) C##a,
enum Color { COLORS };
#undef X #define X(a) #a,
static char *ColorStrings[] = { COLORS };
#undef X
一个真实的例子是在C++编译器 Digital Mars 前端:
#define ENUMSCMAC \
X(unde, SCEXP|SCKEP|SCSCT ) \
X(auto, SCEXP|SCSS|SCRD ) \
X(static, SCEXP|SCKEP|SCSCT) \
X(thread, SCEXP|SCKEP ) \
...
3个独立但并行构造的构建 - 枚举,用于打印的字符串表,以及数组。
我使用过的最复杂的X宏有6个参数,它可以构造枚举,结构初始化,运行时初始化等。
当然,你可能已经在使用一个叫X的宏或者变量,且在宏内部X是硬编码的。
Andrei Alexandrescu(Author of Modern C++ Design)建议以下改进,即将X宏作为参数:
#define FOR_ALL_COLORS(apply) \
apply(red) \
apply(blue) \
apply(green)
紧接着:
#define SELECT_STRING(a) #a,
static char *ColorStrings[] =
{
FOR_ALL_COLORS(SELECT_STRING)
};
#undef SELECT_STRING
任何语言只要支持文本宏预处理程序,X宏技术就能大展身手。
C语言肯定能胜任工作。使用并且传播它,就像我贴心的朋友把他告诉我一样。:)
就像之前说的那样,我还从来没看见过其他人使用这个技巧。因为它晦涩难懂么?欢迎评论。
X Macro的更多相关文章
- FreeMarker学习(宏<#macro>的使用)
原文链接:https://my.oschina.net/weiweiblog/blog/506301?p=1 用户定义指令-使用@符合来调用 有两种不同的类型:Macro(宏)和transform( ...
- configure.ac:32: error: possibly undefined macro: AC_DEFINE
在ubuntu 下编译snappy时,在检查依赖关系时,处理autoconf的包时,在相关依赖包都已经安装的情况下,报如下错误,死活不过. configure.ac:32: error: possib ...
- 【freemaker】之自定义指令<#macro>
测试代码 @Test public void test07(){ try { root.put("name", "张三"); freemakerUtil.fpr ...
- C++ macro(宏)使用小结
谈起C++中的宏,我们第一个想到的应该就是“#define”,它的基本语法长得像这样: #define macroname(para1, para2, para3, ... ,paran) macro ...
- Macro and SQL
If you’ve developed anything in the supply chain area, you’ve most probably come across InventDimJoi ...
- The difference between macro and function I/Ofunction comparision(from c and pointer )
macro is typeless and execute faster than funtion ,becaus of the overhead of calling and returnning ...
- Cmockery macro demo hacking
/********************************************************************* * Cmockery macro demo hacking ...
- __KERNEL__ macro
转载:http://blog.csdn.net/kasalyn/article/details/17097639 The __KERNEL__ macro is defined because the ...
- 幾種方法實現C語言Macro for debug
1. #include <stdio.h> #include <stdlib.h> #define DEBUG 1 #ifdef DEBUG #define DEBUG_PRI ...
- jinja2 宏的简单使用总结(macro)
Table of Contents 1. 简介 2. 用法 3. 参数和变量 4. 注意事项 4.1. macro的变量只能为如下三种: 4.2. 和block的关系: 5. 参考文档 1 简介 ji ...
随机推荐
- 【转】【Oracle 集群】ORACLE DATABASE 11G RAC 知识图文详细教程之RAC 工作原理和相关组件(三)
原文地址:http://www.cnblogs.com/baiboy/p/orc3.html 阅读目录 目录 RAC 工作原理和相关组件 ClusterWare 架构 RAC 软件结构 集群注册(OC ...
- JS 封装一个求圆面积的函数 传值:半径
y(6) var s = ""; function y (r){ s = Math.PI*r*r; alert(s); }
- Django_学生管理系统
一. Django简易学生管理系统 1.在pycharm中创建工程student_manage_system,添加app:student_manage 2.配置静态文件:在工程项目目录下新建目录sta ...
- JavaScript 三要素
一个完整的JavaScript 实现由3部分组成: ECMACcript ECMAScript 规定了这门语言的下列组成部分: 语法 类型 语句 关键字.保留字 操作符 对象为什么要使用DOM? ...
- jquery bind 传参数
方法一. ? 1 2 3 4 function GetCode(event) { alert(event.data.foo); } ? 1 2 3 4 $(document).ready(functi ...
- Linux菜鸟成长日记 ( Linux 下的 ftp 文件传输协议 )
https://blog.csdn.net/buster_zr/article/details/80244542 FTP FTP 是 File Transfer Protocol (文件传输协议)的英 ...
- linux双网卡配置
一.VM虚拟机添加一个网络适配器. 选择自己需要的模式类型 二.启动虚拟机,配置网卡 按原先配置网卡的方式配置完(ip地址及默认网关还有网卡名不能跟原先的一样) 重启所有网卡(service netw ...
- OA项目知识总结2
BaseAction的抽取 项目中的每个实体类都对应一个action 每个action都都要继承ActionSupport类 已以及实现ModelDriver接口 并且需要注入service 虽然 ...
- Spring JDBC模板类—org.springframework.jdbc.core.JdbcTemplate(转)
今天看了下Spring的源码——关于JDBC的"薄"封装,Spring 用一个Spring JDBC模板类来封装了繁琐的JDBC操作.下面仔细讲解一下Spring JDBC框架. ...
- [MongoDB实战]Part1 起步
本书的这部分对MongoDB进行了一个大致的简介.包括了Javascript Shell和Ruby驱动,这俩都有例子 在第一章,我们将了解到MongoDB的历史,设计目的和实际使用的场景.我们还将了解 ...