c中#与##的应用思考
c中#与##的应用思考
- 927
一. 思考出处
在读<<linux 0.12完全剖析>>初始化部分, init进程是通过fork调用的,在这里fork调用的非常特别,由于种种原因,用的是内嵌汇编的方式
- #define _syscall0(type, name) \
- type name(void) \
- { \
- long _res;
- _asm_ volatile ( \
- "int $0x80\n\t" \
- : "=a" (_res) \
- : "0"(__NR_##name); \
- ); \
- if(_res >= 0) \
- return (type) _res; \
- errno = _res; \
- return -1; \
- }
上面看到 _NR_##name, 一下子懵掉了,啥意思这是?
二.思考成果
1.代码解释
不得不说,上面的调用fork方式写的真让我震撼,简单分析一下
前提 : 声明函数 static inline _syscall0(int, fork) //这里 syscall0中的0代表的是没有输入参数的意思,当然1就代表有一个输入参数
过程: ①预编译阶段: //(就是代码还没有编译之前)
static inline _syscall0(int, fork) 将被替换为下面代码
- static inline int fork(void) \
- { \
- long _res;
- _asm_ volatile ( \
- "int $0x80\n\t" \ //系统调用
- : "=a" (_res) \ //将结果存入寄存器%eax, 并且传给 变量 _res
- : "0"(_NR_fork); \ //这里 _NR_fork 在unstd.h中 被#deine 为2, 这里将 _NR_fork 传给寄存器 %eax, 为系统调用做准备
- ); \
- if(_res >= 0) \
- return (type) _res; \
- errno = _res; \
- return -1; \
- }
②运行阶段:
经编译之后呢, 其实static _sys..那个声明其实对fork函数进行说明并且定义了,所以在程序中直接用 fork() 就可以了
模拟的小例子:
- #include <stdio.h>
- #define _NR_hello 10
- #define SUM(a) \
- int a() \
- { \
- return _NR_##a + 2; \
- }
- static SUM( hello); //这里预编译之后编程下面代码了
- //int hello()
- //{
- // return _NR_hello + 2; //由于_NR_hello 前面是有定义的,所以直接替换成 10了
- //}
- int main()
- {
- int c;
- c=hello(); //程序中直接调用hello就可以了
- printf("[%d]\n", c);
- return 0;
- }
以下的参考: http://zgmgypb.blog.163.com/blog/static/9620281920129145154297/
2.#与##
其实就一句话 #,是声明后面定义的是一个字符串,##是告诉编译器,预编译的时候将 ##前后定义的量和在一起
举例: #
- #define MA_IF(EXP) \
- do{ \
- if(EXP) \
- fprintf(stderr, "Warning:" #EXP "\n"); \
- } while(0)
那么实际使用中会出现下面所示的替换过程:
MA_IF (divider == 0);
被替换为
do {
if (divider == 0)
fprintf(stderr, "Warning" "divider == 0" "/n");
} while(0);
其实就是 告诉编译器 EXP 是字符串
举例: ##
这里 ## 的应用可作为代码生成器的编写,这里也是受网上材料的启发,下面的例子就是关于加,减指令以及对应处理函数的关联写法,这样写的好处是提高代码密度以及
便于理解 //由于本人菜鸟,这点体会的很别扭
- #include <stdio.h>
- struct Arithmetic{
- char *action_name;
- int (*func)(int a, int b);
- };
- int add_function(int a, int b)
- {
- return a+b;
- }
- int dev_function(int a, int b)
- {
- return a-b;
- }
- #define FUNC_REGISTER(name) {#name, name##_function} //这里一定要注意一下 不能写成 {name , name##_function},原因下面解释 ①
- int main()
- {
- int data1, data2;
- struct Arithmetic compute[2] = {
- FUNC_REGISTER(add), // ②
- FUNC_REGISTER(dev)
- };
- data1 = compute[0].func(1,2);
- data2 = compute[1].func(2,1);
- printf("[%d][%d]\n", data1, data2);
- }
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中#与##的应用思考的更多相关文章
- 关于《Head First Python》一书中print_lol()函数的思考
关于<Head First Python>一书中print_lol()函数的思考 在<Head First Python>第一章中,讲述到Python处理复杂数据(以电影数据列 ...
- 简述C#中IO的应用 RabbitMQ安装笔记 一次线上问题引发的对于C#中相等判断的思考 ef和mysql使用(一) ASP.NET/MVC/Core的HTTP请求流程
简述C#中IO的应用 在.NET Framework 中. System.IO 命名空间主要包含基于文件(和基于内存)的输入输出(I/O)服务的相关基础类库.和其他命名空间一样. System.I ...
- 关于HashMap中hash()函数的思考
关于HashMap中hash()函数的思考 JDK7中hash函数的实现 static int hash(int h) { h ^= (h >>> 20) ^ (h >&g ...
- 关于在框架中使用curl的思考,以及,curl其实很好用
初步猜想: 在接触到框架文档的第一阶段时,会觉得控制器调用模型就是一件很简单的事,tp中用D方法或者M方法来实例化模型,laravel中用命名空间来加载模型,CI中用$this->load-&g ...
- Android研发中对String的思考(源代码分析)
1.经常使用创建方式思考: String text = "this is a test text "; 上面这一句话实际上是运行了三件事 1.声明变量 String text; ...
- php中的session过期思考一二
看了php开发组成员鸟哥的一篇关于php设置session过期(http://www.laruence.com/2012/01/10/2469.html)的文章 他也说了一般人的回答的几个答案, 回答 ...
- Libgdx中TextButton的一些思考
版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/caihongshijie6/article/details/37566183 由于有 ...
- 005_针对于go语言中速率限制的思考
在之前的go语言的速率限制这篇文章里,我们尝试了普通的速率限制,和脉冲型速率限制.其中,脉冲型速率限制是放开了限制,里面有3个请求是一次性到达,然后再按照200ms的速度限制的,之前的代码如下所示: ...
- 浏览器对localstorage的支持情况以及localstorage在saas系统中的应用实践思考
首先,还是要说,任何一种新特性的引入,通常有着其特有的场景和解决的目标需求,localstorage也一样.在我们的应用场景中,主要在金融业务服务的saas系统.其中涉及很多更改频率很多的元数据的客户 ...
随机推荐
- ztree的CheckBox不显示问题解决办法
问题: 在使用ztree插件时需要设置 zTree 的节点上是否显示 checkbox / radio,但设置后不显示复选框/单选框,如下图所示 设置方法: var setting = { check ...
- python爬虫之路——变量和变量类型
变量类型: ①单值:int ②多值:数组 ③复杂:类 变量类型:就是变量的数据结构,表示这个变量所代表的内容的格式是怎样的. (多值)四种基本数据结构: 列表,字典,元组,集合 列表: ①元素可变, ...
- 第008课_第1个ARM落版程序及引申
form:第008课_第1个ARM裸板程序及引申 第001节_辅线1_硬件知识_LED原理图 当我们学习C语言的时候,我们会写个Hello程序.那当我们下ARM程序,也该有一个简单的程序引领我们入门, ...
- CentOS 软RAID5
yum install -y mdadm np1回车 tfdw mdadm --create /dev/md5 --level=5 --raid-devices=3 /dev/sdb1 /dev/sd ...
- lca(最近公共祖先(在线)) 倍增法详解
转自大佬博客 : https://blog.csdn.net/lw277232240/article/details/72870644 描述:倍增法用于很多算法当中,通过字面意思来理解 LCA是啥呢 ...
- SC || Chapter 1
第一章的重中之重就是这张图吧 (具体参见笔记) ┉┉∞ ∞┉┉┉┉∞ ∞┉┉┉∞ ∞┉┉┉┉∞ ∞┉┉┉┉∞ ∞┉┉┉∞ ∞┉┉┉┉∞ ∞┉┉┉┉∞ ∞┉┉┉∞ ∞┉┉ 区分哪些属性是外部的(面向用户 ...
- ORACLE的SQL JOIN方式大全
ORACLE的SQL JOIN方式大全 在ORACLE数据库中,表与表之间的SQL JOIN方式有多种(不仅表与表,还可以表与视图.物化视图等联结),官方的解释如下所示 A join is a que ...
- 浅谈JavaScript中的正则表达式(适用初学者观看)
浅谈JavaScript中的正则表达式 1.什么是正则表达式(RegExp)? 官方定义: 正则表达式是一种特殊的字符串模式,用于匹配一组字符串,就好比用模具做产品,而正则就是这个模具,定义一种规则去 ...
- 【最大权闭合子图 最小割】bzoj1497: [NOI2006]最大获利
最大权闭合子图的模型:今天才发现dinic板子是一直挂的…… Description 新的技术正冲击着手机通讯市场,对于各大运营商来说,这既是机遇,更是挑战.THU集团旗下的CS&T通讯公司在 ...
- CentOS7下Mysql5.7主从数据库配置
本文配置主从使用的操作系统是Centos7,数据库版本是mysql5.7. 准备好两台安装有mysql的机器(mysql安装教程链接) 主数据库配置 每个从数据库会使用一个MySQL账号来连接主数据库 ...