c语言中#和##的用法
2017-07-10 09:36
393 查看
本文主要讲述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
二、##可以嵌套吗?
看下面的例子:
#define cat(x,y)x##y
宏调用cat(var,123)讲生成var123.
但是,宏调用cat(cat(1,2),3)没有定义:##阻止了外层调用的参数的扩展。因此,它将生成下列的记号串:
cat(1,2)3.
如果要再引入第二层的宏定义,如下定义:
#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)
跟踪宏的展开。
BUILDIO_MEM(b,u8)
BUILDIO_MEM(w,u16)
BUILDIO_MEM(l,u32)
BUILDIO_MEM(q,u64)
就会生成了如下四个函数:
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的配置,都可以找到)。
是的,内核代码有时候这样的调试法……
转自:https://wenku.baidu.com/view/b615bd64be1e650e52ea9939.html
#,##分别在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
二、##可以嵌套吗?
看下面的例子:
#define cat(x,y)x##y
宏调用cat(var,123)讲生成var123.
但是,宏调用cat(cat(1,2),3)没有定义:##阻止了外层调用的参数的扩展。因此,它将生成下列的记号串:
cat(1,2)3.
如果要再引入第二层的宏定义,如下定义:
#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)
跟踪宏的展开。
BUILDIO_MEM(b,u8)
BUILDIO_MEM(w,u16)
BUILDIO_MEM(l,u32)
BUILDIO_MEM(q,u64)
就会生成了如下四个函数:
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的配置,都可以找到)。
是的,内核代码有时候这样的调试法……
转自:https://wenku.baidu.com/view/b615bd64be1e650e52ea9939.html
相关文章推荐
- c语言中的#与##的用法简介
- C语言中的#跟##用法
- C语言宏定义中#与##的用法
- C语言中#和##的用法
- C语言中#和##的用法
- C++基础:C语言中##的用法
- c语言#和## 的用法
- C语言中#和##的用法
- C语言中##和#的用法
- C语言中如何使用宏连接多个字符串(#和##的用法)
- C语言中##的用法以及##在变参的设计
- C语言:宏定义的#和##用法
- C语言宏中#和##的用法 -- 2010.10.30高通笔试
- C语言中##的用法
- C语言中#和##的用法
- C语言中##和#的用法
- c语言define中##用法---ffmpeg
- C语言中可变参数的用法(转载)
- C语言中可变参数的用法(转)
- 关于C语言中的malloc和free函数的用法