C语言中#和##的用法
合集下载
相关主题
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
#define xcat(x,y)cat(x,y) 那么 xcat(xcat(1,2),3)将生成 123, 这是因为 xcat 自身的扩展不包含##运算 符。
三、linux 内核中例子
因为是做 mips 架构的,所以以 mips 为例子。 Linux2.6.25 内核,include/asm-mips/io.h 文件。拷贝一部分的代码出来。 #define__BUILD_MEMORY_SINGLE(pfx,bwlq,type,irq)\ \ staticinlinevoidpfx##write##bwlq(typeval,\ volatilevoid__iomem*mem)\ {\ volatiletype*__mem;\ type__val;\ \ __mem=(void*)__swizzle_addr_##bwlq((unsignedlong)(mem));\ \ __val=pfx##ioswab##bwlq(__mem,val);\ \ if(sizeof(type)!=sizeof(u64)||sizeof(u64)==sizeof(long))\ *__mem=__val;\ /*在这里省略了一些代码*/ } #define__BUILD_MEMORY_PFX(bus,bwlq,type)\ \ __BUILD_MEMORY_SINGLE(bus,bwlq,type,1) #defineBUILDIO_MEM(bwlq,type)\ \ __BUILD_MEMORY_PFX(__raw_,bwlq,type)\ __BUILD_MEMORY_PFX(,bwlq,type)\ __BUILD_MEMORY_PFX(__mem_,bwlq,type)\ BUILDIO_MEM(b,u8) BUILDIO_MEM(w,u16) BUILDIO_MEM(l,u32) BUILDIO_MEM(q,u64) 跟踪宏的展开。
本文主要讲述 c 语言的一点基础语法和在内核的应用中其中的一点例子。 #,##分别在 c 语言中是怎么作用? 文章代码编译的环境: 桌面环境:Ubuntu10.04 内核:linux2.6.32 编译器:gcc4.4.3 一、基本的用法 1、#.参数名以#作为前缀则结果将被扩展为由实际参数的带引号的字符串。 如: #define dprint(expr)printf(#expr"=%d\n",expr); intmain() { inta=20,b=10; dprint(a/b); return0; } 上面的例子会打印出: a/b=2 2、##.预处理器运算符##为宏提供了一种连接实际参数的手段。如果替换文本中 的参数与##相邻,则该参数将被实际参数替换,##与前后的空白将被删除,并对 替换后的结果重新扫描。 形成一个新的标号,如果这样产生的记号无效,或者结果依赖于##运算顺序,则 结果没有定义。 如: #definepaste(front,back)front##back 因此,宏调用 paste(name,_xiaobai)的结果为 name_xiaobai. 如: #define createfun(name1,name2)\ void name1##name2()\ {\ printf("%scalled\n",__FUNCTION__);\ } createfun(the,function); intmain() { thefunction(); return0; } 输出的结果是:thefunctioncalled
BUILDIO_MEM(b,u8) BUILDIO_MEM(w,u16) BUILDIO_MEM(l,u32) BUILDIO_MEM(q,u64)
wenku.baidu.com 就会生成了如下四个函数:
staticinlinevoidwriteb(u8val,volatilevoid__iomem*mem){……} staticinlinevoidwritew(u16val,volatilevoid__iomem*mem){……} staticinlinevoidwritel(u32val,volatilevoid__iomem*mem){……} staticinlinevoidwriteq(u64val,volatilevoid__iomem*mem){……} 同时,如果当我们用函数类似 writeb 之类的出现了问题,一般情况下用编辑工具 是找不到函数定义的,于是乎跟踪不是去了,其实不然,可以针对里面的关键字,
譬如:write.是可以找到的(在 linux 下面用 find,grep,或者是 vim 的配置,都可以 找到)。
是的,内核代码有时候这样的调试法……
二、##可以嵌套吗? 看下面的例子:
#define cat(x,y)x##y
宏调用 cat(var,123)讲生成 var123. 但是,宏调用 cat(cat(1,2),3)没有定义:##阻止了外层调用的参数的扩展。 因此,它将生成下列的记号串:
cat(1,2)3. 如果要再引入第二层的宏定义,如下定义:
三、linux 内核中例子
因为是做 mips 架构的,所以以 mips 为例子。 Linux2.6.25 内核,include/asm-mips/io.h 文件。拷贝一部分的代码出来。 #define__BUILD_MEMORY_SINGLE(pfx,bwlq,type,irq)\ \ staticinlinevoidpfx##write##bwlq(typeval,\ volatilevoid__iomem*mem)\ {\ volatiletype*__mem;\ type__val;\ \ __mem=(void*)__swizzle_addr_##bwlq((unsignedlong)(mem));\ \ __val=pfx##ioswab##bwlq(__mem,val);\ \ if(sizeof(type)!=sizeof(u64)||sizeof(u64)==sizeof(long))\ *__mem=__val;\ /*在这里省略了一些代码*/ } #define__BUILD_MEMORY_PFX(bus,bwlq,type)\ \ __BUILD_MEMORY_SINGLE(bus,bwlq,type,1) #defineBUILDIO_MEM(bwlq,type)\ \ __BUILD_MEMORY_PFX(__raw_,bwlq,type)\ __BUILD_MEMORY_PFX(,bwlq,type)\ __BUILD_MEMORY_PFX(__mem_,bwlq,type)\ BUILDIO_MEM(b,u8) BUILDIO_MEM(w,u16) BUILDIO_MEM(l,u32) BUILDIO_MEM(q,u64) 跟踪宏的展开。
本文主要讲述 c 语言的一点基础语法和在内核的应用中其中的一点例子。 #,##分别在 c 语言中是怎么作用? 文章代码编译的环境: 桌面环境:Ubuntu10.04 内核:linux2.6.32 编译器:gcc4.4.3 一、基本的用法 1、#.参数名以#作为前缀则结果将被扩展为由实际参数的带引号的字符串。 如: #define dprint(expr)printf(#expr"=%d\n",expr); intmain() { inta=20,b=10; dprint(a/b); return0; } 上面的例子会打印出: a/b=2 2、##.预处理器运算符##为宏提供了一种连接实际参数的手段。如果替换文本中 的参数与##相邻,则该参数将被实际参数替换,##与前后的空白将被删除,并对 替换后的结果重新扫描。 形成一个新的标号,如果这样产生的记号无效,或者结果依赖于##运算顺序,则 结果没有定义。 如: #definepaste(front,back)front##back 因此,宏调用 paste(name,_xiaobai)的结果为 name_xiaobai. 如: #define createfun(name1,name2)\ void name1##name2()\ {\ printf("%scalled\n",__FUNCTION__);\ } createfun(the,function); intmain() { thefunction(); return0; } 输出的结果是:thefunctioncalled
BUILDIO_MEM(b,u8) BUILDIO_MEM(w,u16) BUILDIO_MEM(l,u32) BUILDIO_MEM(q,u64)
wenku.baidu.com 就会生成了如下四个函数:
staticinlinevoidwriteb(u8val,volatilevoid__iomem*mem){……} staticinlinevoidwritew(u16val,volatilevoid__iomem*mem){……} staticinlinevoidwritel(u32val,volatilevoid__iomem*mem){……} staticinlinevoidwriteq(u64val,volatilevoid__iomem*mem){……} 同时,如果当我们用函数类似 writeb 之类的出现了问题,一般情况下用编辑工具 是找不到函数定义的,于是乎跟踪不是去了,其实不然,可以针对里面的关键字,
譬如:write.是可以找到的(在 linux 下面用 find,grep,或者是 vim 的配置,都可以 找到)。
是的,内核代码有时候这样的调试法……
二、##可以嵌套吗? 看下面的例子:
#define cat(x,y)x##y
宏调用 cat(var,123)讲生成 var123. 但是,宏调用 cat(cat(1,2),3)没有定义:##阻止了外层调用的参数的扩展。 因此,它将生成下列的记号串:
cat(1,2)3. 如果要再引入第二层的宏定义,如下定义: