c中#与##的应用思考

原创 2014年02月25日 22:01:35
  • 927

一. 思考出处

在读<<linux 0.12完全剖析>>初始化部分, init进程是通过fork调用的,在这里fork调用的非常特别,由于种种原因,用的是内嵌汇编的方式

  1. #define _syscall0(type, name) \
  2. type name(void) \
  3. { \
  4. long _res;
  5. _asm_ volatile ( \
  6. "int $0x80\n\t"  \
  7. : "=a" (_res) \
  8. : "0"(__NR_##name);    \
  9. ); \
  10. if(_res >= 0) \
  11. return (type) _res; \
  12. errno = _res; \
  13. return -1;   \
  14. }

上面看到 _NR_##name, 一下子懵掉了,啥意思这是?

二.思考成果

1.代码解释

不得不说,上面的调用fork方式写的真让我震撼,简单分析一下

前提 : 声明函数 static inline _syscall0(int, fork)   //这里  syscall0中的0代表的是没有输入参数的意思,当然1就代表有一个输入参数

过程:  ①预编译阶段:        //(就是代码还没有编译之前)

static inline _syscall0(int, fork)  将被替换为下面代码

  1. static inline int fork(void) \
  2. { \
  3. long _res;
  4. _asm_ volatile ( \
  5. "int $0x80\n\t"  \     //系统调用
  6. : "=a" (_res) \        //将结果存入寄存器%eax, 并且传给 变量 _res
  7. : "0"(_NR_fork);    \  //这里 _NR_fork 在unstd.h中 被#deine 为2, 这里将 _NR_fork 传给寄存器 %eax, 为系统调用做准备
  8. ); \
  9. if(_res >= 0) \
  10. return (type) _res; \
  11. errno = _res; \
  12. return -1;   \
  13. }

②运行阶段:

经编译之后呢, 其实static _sys..那个声明其实对fork函数进行说明并且定义了,所以在程序中直接用 fork() 就可以了

模拟的小例子:

  1. #include <stdio.h>
  2. #define _NR_hello 10
  3. #define SUM(a) \
  4. int a() \
  5. {  \
  6. return _NR_##a + 2; \
  7. }
  8. static SUM( hello); //这里预编译之后编程下面代码了
  9. //int hello()
  10. //{
  11. //  return _NR_hello + 2; //由于_NR_hello 前面是有定义的,所以直接替换成 10了
  12. //}
  13. int main()
  14. {
  15. int c;
  16. c=hello(); //程序中直接调用hello就可以了
  17. printf("[%d]\n", c);
  18. return 0;
  19. }

以下的参考: http://zgmgypb.blog.163.com/blog/static/9620281920129145154297/

2.#与##

其实就一句话 #,是声明后面定义的是一个字符串,##是告诉编译器,预编译的时候将 ##前后定义的量和在一起

举例: #

  1. #define MA_IF(EXP) \
  2. do{ \
  3. if(EXP) \
  4. fprintf(stderr, "Warning:" #EXP "\n"); \
  5. } while(0)

那么实际使用中会出现下面所示的替换过程:

MA_IF (divider == 0);

被替换为

do {
  if (divider == 0)
  fprintf(stderr, "Warning" "divider == 0" "/n");
} while(0);

其实就是 告诉编译器 EXP 是字符串

举例: ##

这里 ##  的应用可作为代码生成器的编写,这里也是受网上材料的启发,下面的例子就是关于加,减指令以及对应处理函数的关联写法,这样写的好处是提高代码密度以及

便于理解 //由于本人菜鸟,这点体会的很别扭

  1. #include <stdio.h>
  2. struct Arithmetic{
  3. char *action_name;
  4. int (*func)(int a, int b);
  5. };
  6. int add_function(int a, int b)
  7. {
  8. return a+b;
  9. }
  10. int dev_function(int a, int b)
  11. {
  12. return a-b;
  13. }
  14. #define FUNC_REGISTER(name) {#name, name##_function} //这里一定要注意一下 不能写成 {name , name##_function},原因下面解释 ①
  15. int main()
  16. {
  17. int data1, data2;
  18. struct Arithmetic compute[2] = {
  19. FUNC_REGISTER(add),  // ②
  20. FUNC_REGISTER(dev)
  21. };
  22. data1 = compute[0].func(1,2);
  23. data2 = compute[1].func(2,1);
  24. printf("[%d][%d]\n", data1, data2);
  25. }

nop: 这里如果写成①这种情况, 那么编译会报错, add,dev没有被定义, 这是为何?
            回答, 不管是#, ##  等等,我们预编译的时候会进行宏替换, 比如,我们#define SUN(a), a    , 如果我们在main程序中 SUN(1)

则会替换成 1,1的话没什么好说的,因为就是常量嘛,但是如果我们写成SUN(abc),那么编译器就替换成abc,但是编译器不认识abc是什么啊,因为你没有告诉它

同理: 这里写成①的情况,在②中进行宏替换,替换成{abc, name_function} ,这里abc没有被定义,编译器不知道是啥,更别说进行赋值操作,而name_function这个是            有定义的, 是个函数!,所以正确,所以我们必须在name前面加#告诉编译器它是个字符串,就可以了

那么在②中进行宏替换之后,就变成了{"add", name_function}了

c中#与##的应用思考的更多相关文章

  1. 关于《Head First Python》一书中print_lol()函数的思考

    关于<Head First Python>一书中print_lol()函数的思考 在<Head First Python>第一章中,讲述到Python处理复杂数据(以电影数据列 ...

  2. 简述C#中IO的应用 RabbitMQ安装笔记 一次线上问题引发的对于C#中相等判断的思考 ef和mysql使用(一) ASP.NET/MVC/Core的HTTP请求流程

    简述C#中IO的应用   在.NET Framework 中. System.IO 命名空间主要包含基于文件(和基于内存)的输入输出(I/O)服务的相关基础类库.和其他命名空间一样. System.I ...

  3. 关于HashMap中hash()函数的思考

    关于HashMap中hash()函数的思考 JDK7中hash函数的实现   static int hash(int h) { h ^= (h >>> 20) ^ (h >&g ...

  4. 关于在框架中使用curl的思考,以及,curl其实很好用

    初步猜想: 在接触到框架文档的第一阶段时,会觉得控制器调用模型就是一件很简单的事,tp中用D方法或者M方法来实例化模型,laravel中用命名空间来加载模型,CI中用$this->load-&g ...

  5. Android研发中对String的思考(源代码分析)

    1.经常使用创建方式思考: String text = "this is a test text "; 上面这一句话实际上是运行了三件事  1.声明变量 String text; ...

  6. php中的session过期思考一二

    看了php开发组成员鸟哥的一篇关于php设置session过期(http://www.laruence.com/2012/01/10/2469.html)的文章 他也说了一般人的回答的几个答案, 回答 ...

  7. Libgdx中TextButton的一些思考

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/caihongshijie6/article/details/37566183         由于有 ...

  8. 005_针对于go语言中速率限制的思考

    在之前的go语言的速率限制这篇文章里,我们尝试了普通的速率限制,和脉冲型速率限制.其中,脉冲型速率限制是放开了限制,里面有3个请求是一次性到达,然后再按照200ms的速度限制的,之前的代码如下所示: ...

  9. 浏览器对localstorage的支持情况以及localstorage在saas系统中的应用实践思考

    首先,还是要说,任何一种新特性的引入,通常有着其特有的场景和解决的目标需求,localstorage也一样.在我们的应用场景中,主要在金融业务服务的saas系统.其中涉及很多更改频率很多的元数据的客户 ...

随机推荐

  1. 宿主机Windows访问虚拟机Linux文件(一)

    如果用户使用windows操作系统,但是在虚拟机下配置Linux内核操作操作系统,往往需要实现通过宿主机Windows操作系统访问Linux内核操作系统中资源.本次实验实现的是宿主机windows 1 ...

  2. HDU 5501 The Highest Mark (贪心+DP,经典)

    题意: 有n道题目,每道题目的初始分数为Ai,分数每分钟减少Bi,完成此题需要Ci分钟,问在t分钟内最多能获得多少分? 思路: 好题~ 如果没有B的话,就是一道裸的01背包的题目了.每道题目的得分为: ...

  3. 51nod 1191 消灭兔子

    题目来源: 2013腾讯马拉松赛第三场 基准时间限制:1 秒 空间限制:131072 KB 分值: 40 难度:4级算法题 有N只兔子,每只有一个血量B[i],需要用箭杀死免子.有M种不同类型的箭可以 ...

  4. 【Python图像特征的音乐序列生成】使用Python生成简单的MIDI文件

    这个全新的Python音乐创作系列,将会不定期更新.写作这个系列的初衷,是为了做一个项目<基于图像特征的音乐序列生成模型>,实时地提取照片特征,进行神经网络处理,生成一段音乐. 千里之行, ...

  5. 关于日志造成的频繁的IO

    记录日志可能消耗大量的IO [Q] 每次写入都是一个IO操作 即使是同一个文件 两次写入也要打开两次IO操作 [F] 设想有这样一个扩展  把php中要记录的日志 用文件名 和 内容的方式记录在内存中 ...

  6. UVA 1608 Non-boring sequence 不无聊的序列(分治,中途相遇)

    题意:给你一个长度为n序列,如果这个任意连续子序列的中都有至少出现一次的元素,那么就称这个序列是不无聊的,判断这个序列是不是无聊的. 先预处理出每个元素之前和之后相同元素出现的位置,就可以在O(1)的 ...

  7. [学习笔记] C++ 历年试题解析(二)--程序题

    发现程序题也挺有价值的. 顺便记录下来几道. 1.题目 #include <iostream> #include <cstring> using namespace ① std ...

  8. tomcat - CPU高占用问题记录

    先查询进程 top 再根据进程号,查出进程的线程 ps p 3036 -L -o pcpu,pid,tid,time,tname,cmd 得到最高使用率CPU的线程TID,转换成16进制 printf ...

  9. c++ sort用法 学习笔记

    c++ sort排序函数,需要加库#include<algorithm>,语法描述:sort(begin,end,cmp),cmp参数可以没有,如果没有默认非降序排序. 首先是升序排序: ...

  10. Maven配置项目依赖使用本地仓库的方法汇总

    Maven配置项目使用本地仓库有以下方式实现: 1.类似本地仓库,但是属于本地依赖,比如某个JAR包是引用第三方的,直接放在了项目的lib文件夹,那么此时可以如下配置项目的POM: <depen ...