转自:http://chongsoft.bokee.com/5816474.html

今天测试了宏定义中的 "#" 和 "##" 的区别。

结果如下:

"#"  代表和一个字符串相连接

"##"  代表和一个符号连接,符号可以是变量,或另一个宏符号。

举例如下:

宏定义如下

(1)

#define DEV_FILE_NAME    "/dev/test_kft"

#define OPEN_FILE(fd, n)    \
{   \
    fd = open(DEV_FILE_NAME #n,O_RDONLY); \
    if(fd < 0)  \
    { \
       printf("Open device error\n");  \
        return 0; \
    }   \
}

如此调用:

OPEN_FILE(fd1, 1);
    OPEN_FILE(fd2, 2);
    OPEN_FILE(fd3, 3);
    OPEN_FILE(fd4, 4);
    OPEN_FILE(fd5, 5);
    OPEN_FILE(fd6, 6);

用gcc -E展开后,如下

2299:    { fd1 = open("/dev/test_kft" "1",00); if(fd1 < 0) { printf("Open device error\n"); return 0; } };
2300:    { fd2 = open("/dev/test_kft" "2",00); if(fd2 < 0) { printf("Open device error\n"); return 0; } };
2301:    { fd3 = open("/dev/test_kft" "3",00); if(fd3 < 0) { printf("Open device error\n"); return 0; } };
2302:    { fd4 = open("/dev/test_kft" "4",00); if(fd4 < 0) { printf("Open device error\n"); return 0; } };
2303:    { fd5 = open("/dev/test_kft" "5",00); if(fd5 < 0) { printf("Open device error\n"); return 0; } };
2304:    { fd6 = open("/dev/test_kft" "6",00); if(fd6 < 0) { printf("Open device error\n"); return 0; } };

如果没有定义DEV_FILE_NAME ,就是

2299:    { fd1 = open(DEV_FILE_NAME "1",00); if(fd1 < 0) { printf("Open device error\n"); return 0; } };
2300:    { fd2 = open(DEV_FILE_NAME "2",00); if(fd2 < 0) { printf("Open device error\n"); return 0; } };
2301:    { fd3 = open(DEV_FILE_NAME "3",00); if(fd3 < 0) { printf("Open device error\n"); return 0; } };
2302:    { fd4 = open(DEV_FILE_NAME "4",00); if(fd4 < 0) { printf("Open device error\n"); return 0; } };
2303:    { fd5 = open(DEV_FILE_NAME "5",00); if(fd5 < 0) { printf("Open device error\n"); return 0; } };
2304:    { fd6 = open(DEV_FILE_NAME "6",00); if(fd6 < 0) { printf("Open device error\n"); return 0; } };

所以可以很清楚的看出#n 解析出来的是"n" , 用于连接一个已有的字符串。

(2)  再来看 ## 是什么意思, 宏定义如下:

#define OPEN_FILE(fd, n)    \
{   \
    fd = open(DEV_FILE_NAME ##n,O_RDONLY); \
    if(fd < 0)  \
    { \
       printf("Open device error\n");  \
        return 0; \
    }   \
}

调用方式相同。

看宏展开:

2299:    { fd1 = open(DEV_FILE_NAME1,00); if(fd1 < 0) { printf("Open device error\n"); return 0; } };
2300:    { fd2 = open(DEV_FILE_NAME2,00); if(fd2 < 0) { printf("Open device error\n"); return 0; } };
2301:    { fd3 = open(DEV_FILE_NAME3,00); if(fd3 < 0) { printf("Open device error\n"); return 0; } };
2302:    { fd4 = open(DEV_FILE_NAME4,00); if(fd4 < 0) { printf("Open device error\n"); return 0; } };
2303:    { fd5 = open(DEV_FILE_NAME5,00); if(fd5 < 0) { printf("Open device error\n"); return 0; } };
2304:    { fd6 = open(DEV_FILE_NAME6,00); if(fd6 < 0) { printf("Open device error\n"); return 0; } };

现在看清楚了, ##n 的作用是把n直接连接在了一个符号的末尾. 好, 现在我们定义一个符号看看效果。

#define DEV_FILE_NAME1    "/dev/test_kft1"

再展开:

2299:    { fd1 = open("/dev/test_kft1",00); if(fd1 < 0) { printf("Open device error\n"); return 0; } };
2300:    { fd2 = open(DEV_FILE_NAME2,00); if(fd2 < 0) { printf("Open device error\n"); return 0; } };
2301:    { fd3 = open(DEV_FILE_NAME3,00); if(fd3 < 0) { printf("Open device error\n"); return 0; } };
2302:    { fd4 = open(DEV_FILE_NAME4,00); if(fd4 < 0) { printf("Open device error\n"); return 0; } };
2303:    { fd5 = open(DEV_FILE_NAME5,00); if(fd5 < 0) { printf("Open device error\n"); return 0; } };
2304:    { fd6 = open(DEV_FILE_NAME6,00); if(fd6 < 0) { printf("Open device error\n"); return 0; } };

很显然第一个符号被替换了, 因为是符号是宏的缘故。 这样我们也能把这一扩展特性应用在变量上。

测试的话,可以用下面的代码来测试。

#include <stdio.h>
#define OPEN(a,b) {a#b} int main()
{
OPEN(cwd,);
OPEN(CWD,);
return ;
}

把其中的#换成#就是另外一个

另外注意 gcc -E 的好处

gnu 扩展之#和##的更多相关文章

  1. Obstack是C标准库里面对内存管理的GNU扩展

    Obstack是C标准库里面对内存管理的GNU扩展 Obstack介绍 Obstack初始化 在Obstack中申请对象 释放对象 申请growing object 获取Obstack状态 数据对齐 ...

  2. GNU C 扩展(转)

    GNU CC 是一个功能非常强大的跨平台 C 编译器,它对 C 语言提供了很多扩展,这些扩展对优化.目标代码布局.更安全的检查等方面提供了很强的支持.这里对支持支持 GNU 扩展的 C 语言成为 GN ...

  3. Linux 内核使用的 GNU C 扩展

    gcc核心扩展linuxforum(转)=========================== Linux 内核使用的 GNU C 扩展 =========================== GNC ...

  4. 关于GNU规范的语法扩展

    GNU 是一款能用于构建类 Unix 操作系统的计算机软件合集,由自由软件之父 Richard Stallman 开创,于 1983 年 9 月 27 日对外发布.GNU 完全由自由软件(free s ...

  5. 感悟 GNU C 以及将 Vim 打造成 C/C++ 的半自动化 IDE

    C 语言在 Linux 系统中的重要性自然是无与伦比.不可替代,所以我写 Linux 江湖系列不可能不提 C 语言.C 语言是我的启蒙语言,感谢 C 语言带领我进入了程序世界.虽然现在不靠它吃饭,但是 ...

  6. PHP扩展开发相关总结

    1.线程安全宏定义 在TSRM/TSRM.h文件中有如下定义 #define TSRMLS_FETCH() void ***tsrm_ls = (void ***) ts_resource_ex(0, ...

  7. 用C/C++扩展你的PHP(转)

    简 介 英文版下载: PHP 5 Power Programming PHP取得成功的一个主要原因之一是她拥有大量的可用扩展.web开发者无论有何种需求,这种需求最有可能在PHP发行包里找到.PHP发 ...

  8. GNU C编译器的gnu11和c11

    国际标准组织发布c11后,gnu为自己的编译器发布两种标准gnu11和c11 gnu11:带gnu c扩展的c11标准,如果你的代码包含了typeof,__attribute__等等gnu的扩展,就必 ...

  9. 用C/C++扩展你的PHP 为你的php增加功能

    英文版下载: PHP 5 Power Programming http://www.jb51.net/books/61020.html PHP取得成功的一个主要原因之一是她拥有大量的可用扩展.web开 ...

随机推荐

  1. BC 2015在百度之星程序设计大赛 - 预赛(1)(矩形区域-旋转卡)

    矩形区域 Accepts: 717 Submissions: 1619 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 ...

  2. javascript变量,作用域和内存问题(一)

          js对象的引用是很有意思的,引用型对象是不可以直接引用的,我猜测这是原型的来源之一,有大神请详解或斧正.    “引用类型的值是保存在内存中的对象.与其他语言不同,JavaScript不允 ...

  3. c语言实现hashtable,相似C++的map和iOS的NSDictionary

    跟线性数组和链表不同.HashTable是高速查找的数据结构.本文中的HashTable使用链表处理数组. 该HashTable能够指定table的长度.提供了遍历的方法. 包含table的长度的选择 ...

  4. Matlab Newton‘s method

    定义函数 function y=f(x) y=f(x).%函数f(x)的表达式 end function z=h(x) z=h(x).%函数h(x)的表达式 end 主程序 x=X;%迭代初值 i=0 ...

  5. 8、Cocos2dx 3.0三,找一个小游戏开发3.0存储器管理的版本号

    重开发人员的劳动成果,转载的时候请务必注明出处:http://blog.csdn.net/haomengzhu/article/details/27693365 复杂的内存管理 移动设备上的硬件资源十 ...

  6. java major version(转)

    在jar包中,用winrar解压一个类文件,然后在命令行下面输入 javap -verbose classname 会输出一些信息,大致如下: Compiled from "HtmlCraw ...

  7. C# Socket TCP Server & Client & nodejs client

    要调试公司某项目里的一个功能,因为要准备测试环境,趁这个机会重温了一下Socket(全还给老师了 -_-#),做个备份. C# Server static void Main(string[] arg ...

  8. Android开发自学笔记(基于Android Studio1.3.1)—1.环境搭建(转)

    一.引言    本套学习笔记的开发环境是Windows 10 专业版和Android Studio 的最新版1.3.1. Android Studio 是一个Android开发环境,基于Intelli ...

  9. 3-08. 栈模拟队列(25)(ZJU_PAT 模拟)

    主题链接:http://pat.zju.edu.cn/contests/ds/3-08 设已知有两个堆栈S1和S2,请用这两个堆栈模拟出一个队列Q. 所谓用堆栈模拟队列,实际上就是通过调用堆栈的下列操 ...

  10. ASP.NET程序读取二代身份证(附源码)

    原文:ASP.NET程序读取二代身份证(附源码) 一般来说winform应用程序解决这个问题起来时很容易的,web应用程序就麻烦一点了. 这里我说说我的解决思路: 一.你必要有联机型居民身份证阅读器一 ...