C语言宏定义##连接符和#符的使用

原文地址:http://blog.csdn.net/vvzaixian/article/details/7067214

 

在学习使用OmapL138时,看到程序中使用很多宏定义,学习一下

宏定义是用一组伪操作来实现的。其格式是:

    macro_name    MACRO    [dumny_parameter_list]

                   ...     (宏定义体)

                  ENDM

其中MACRO和ENDM是一对伪操作.这对伪操作之间是宏定义体--是一组独立功能的程序代码.宏指令名(macro_name)给出该宏定义的名称,调用时就使用宏指令名来调用该宏定义.其中哑元表(dumny_parameter_list)给出了该宏定义中所用到的形式参数(或称虚参),每个哑元之间用逗号隔开.

    经宏定义后的宏指令就可以在源程序中调用.这种对宏指令的调用称为宏调用,宏调用的格式是:

macro name [actual parameter list]

实元表(actual_parameter_list)中的每一项为实元,相互之间用逗号隔开

 

 ##连接符号由两个井号组成,其功能是在带参数的宏定义中将两个子串(token)联接起来,从而形成一个新的子串。但它不可以是第一个或者最后一个子串。所谓的子串(token)就是指编译器能够识别的最小语法单元。

    简单的说,“##”是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。其中,分隔的作用类似于空格。我们知道在普通的宏定义中,预处理器一般把空格解释成分段标志,对于每一段和前面比较,相同的就被替换。但是这样做的结果是,被替换段之间存在一些空格。如果我们不希望出现这些空格,就可以通过添加一些##来替代空格。

举列比较下述几个宏定义的区别

   #define A1(name, type)  type name_##type##_type 或

   #define A2(name, type)  type name##_##type##_type

   A1(a1, int);  /* 等价于: int name_int_type; */

   A2(a1, int);  /* 等价于: int a1_int_type;   */

   解释:

        1) 在第一个宏定义中,”name”和第一个”_”之间,以及第2个”_”和第二个 ”type”之间没有被分隔,所以预处理器会把name_##type##_type解释成3段:“name_”、“type”、以及“_type”,这中间只有“type”是在宏前面出现过的,所以它可以被宏替换。

        2) 而在第二个宏定义中,“name”和第一个“_”之间也被分隔了,所以预处理器会把name##_##type##_type解释成4段:“name”、“_”、“type”以及“_type”,这其间,就有两个可以被宏替换了。

        3) A1和A2的定义也可以如下:

           #define A1(name, type)  type name_  ##type ##_type 

                                      <##前面随意加上一些空格>

           #define A2(name, type)  type name ##_ ##type ##_type

    结果是##会把前面的空格去掉完成强连接,得到和上面结果相同的宏定义。或再比如

#define LINK_MULTIPLE(a,b,c,d) a##_##b##_##c##_##d

typedef struct _record_type LINK_MULTIPLE(name,company,position,salary);

// 这里这个语句将展开为:

// typedef struct _record_type name_company_position_salary;

其他相关 – 单独的一个 #

   至于单独一个#,则表示对这个变量替换后,再加双引号引起来。比如

      #define  __stringify_1(x)   #x

那么

      __stringify_1(linux)   <==>  ”linux”

所以,对于MODULE_DEVICE_TABLE

     1) #define MODULE_DEVICE_TABLE(type,name)                       

             MODULE_GENERIC_TABLE(type##_device,name)

     2) #define MODULE_GENERIC_TABLE(gtype,name)                     

             extern const struct gtype##_id __mod_##gtype##_table    

             __attribute__ ((unused, alias(__stringify(name))))

得到 

      MODULE_DEVICE_TABLE(usb, products) 

                             /*notes: struct usb_device_id products; */

 <==> MODULE_GENERIC_TABLE(usb_device,products)

 <==> extern const struct usb_device_id __mod_usb_device_table    

             __attribute__ ((unused, alias(”products”)))  

注意到alias attribute需要一个双引号,所以在这里使用了__stringify(name)来给name加上双引号。另外,还注意到一个外部变量”__mod_usb_device_table”被alias到了本驱动专用的由用户自定义的变量products<usb_device_id类型>。

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值