c语言中_的用法,c语言中#和##的用法

本文介绍了C语言预处理器中的##和#两个符号的用法。#用于将参数转换为带引号的字符串,##用于连接参数。在Linux内核中,这两个符号被巧妙地应用于宏定义,例如在内存操作函数的生成中,通过宏展开创建了如writeb、writew等函数。通过这种方式,可以动态生成针对不同数据类型的函数,简化了代码并提高了效率。
摘要由CSDN通过智能技术生成

本文主要讲述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的配置,都可以找到)。

是的,内核代码有时候这样的调试法……

(学习的目的是增长知识,提高能力,相信一分耕耘一分收获,努力就一定可以获得应有的回报)

7 / 7下载文档可编辑

展开阅读全文

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值