zoukankan      html  css  js  c++  java
  • #和##在宏替换中的作用

    #include <stdio.h>
    #define f(a,b) a##b
    #define g(a)  #a
    #define h(a) g(a)
    int main()
    {
      printf("%s/n",h(f(1,2)));
      printf("%s/n",g(f(1,2)));
      return 0;
    }

    首先需要了解#和##的意义。
     
    #  将右边的参数做整体的字符串替换。
     
    #define g(a)  #a
     
    则g(hello world) à hello world; g(sleep(1)) à sleep(1)
     
    对于#的参数,即便是另一个宏,也不展开,仍然作为字符串字面信息输出。
     
    所以,g(f(1,2)) à f(1,2)
     
    对于h(f(1,2)),由于h(a)是非#或##的普通宏,需要先宏展开其参数a,即展开f(1,2)为12,则h(a) 宏替换为h(12),进而宏替换为g(12), 进而宏替换为12 
     
    ## 将左右两边的参数做整体的字符串拼接替换。
     
    #define f(a,b) a##b
     
    则f(1,2) à 12, f(i,1) à i1
     
    同#,对于##的参数,即便是另一个宏,也不展开,仍然作为字符串字面信息输出。
     
    此外,有一个限制是,经过##替换后的内容必须能够作为一个合法的变量。
     
    以上f(i,1) à i1中,如果程序中没有i1的定义,或者通过f(1,i)构成1i,则即便是通过了宏替换,也不能编译通过。

    --------------------------------------------------------------------------------

    日常实践中,##是常用的替换。尤其在通过C的函数指针来模拟动态绑定时大有用处。
     
    下面是从stackoverflow(具体id记不清了)上摘取的一个例子。

    struct command
    {
      char *name;
      void (*function) (void);
    };
    struct command commands[] =
    {
      { "quit", quit_command },
      { "help", help_command },
      ...
    };

    构造一个这样的commands数组,是为了在后续的设计中,可以通过交互式的字符串输入,来动态地执行相应的函数。字符串自身作为前缀,”_command”作为后缀。

    手工构造这样第一个commands数组显得非常冗余,还容易造成不必要的拼写错误。下面来看看这个宏替换版本。

    #define COMMAND(NAME)  { #NAME, NAME ## _command } 
    struct command commands[] =
    {
    COMMAND (quit),
    COMMAND (help),
    ...
    };

    清晰了很多!

  • 相关阅读:
    [转]PostgreSQL数据类型
    Linux下执行自定义的可执行命令无效原因
    [其它]iOS 12.2支持电信VoLTE了,中国电信教你如何开通:只要三步
    本机无法访问虚拟机里面的nginx的80端口
    百度的网络接入架构图
    如何让局域网中的其他主机访问虚拟机
    java中synchronized 用在实例方法和对象方法上面的区别
    Redis登陆服务器和批量删除指定的key
    vim查找关键字的好方法
    网络攻防之动态修改表单的值
  • 原文地址:https://www.cnblogs.com/catgatp/p/8604245.html
Copyright © 2011-2022 走看看