C/C++ 的宏中:

  (1) # 的功能是将其后面的宏参数进行字符串化操作,简单说就是在对它所引用的宏变量通过替换后在其左右各加上一个双引号。

  也就是说:

  #define __TO_STRING_IMPL(x) #x 

  中的#X, 则宏变量X所引用的是宏参数中的X,因为,#X会直接对参数X进行字符串化,因此该参数输出的是字符串“X”。

  使用: __TO_STRING_IMPL(15), 输出:“15”

  为了能够进一步增加宏的参数输入,因此需要在外部再包括一个宏转化,这样就可以把"转化宏"中的具体参数传入到参数化宏中,这样就可以变为相应“具体参数”的字符串。

  如下图所示:

  #define __TO_STRING(x) __TO_STRING_IMPL(x)

  (2) ##连接符号由两个井号组成,其功能是在带参数的宏定义中将两个子串联接起来,从而形成一个新的子串。但它不可以是第一个或者最后一个子串。

  作用:只是用所引用的宏变量替换后,将两个字符串串接起来,形成新的字符串。

  如下:

  #define A(x) T_##x

  使用:A(1),则输出:T_1

  

  (3)#@, 表示将参数作为字符形式替换

  如下:

  #define B(x) #@x

  使用: B(1),输出:’1‘

凡是宏定义里有用#或##的地方宏参数是不会再展开,例如_STRI(INT_MAX)中的INT_MAX就不会被展开为2147483647。  

如果想要使其中的宏参数展开,则需要多加一层中间转换宏: #define STRI(s) _STRI(s)  

加这层宏的用意是把所有宏的参数在这层里全部展开,那么在转换宏里的宏就能得到对应的宏参数。

  

#ifndef __TO_STRING
#define __TO_STRING_IMPL(x) #x
#define __TO_STRING(x) __TO_STRING_IMPL(x)
#endif

  例子如下:

#include "stdafx.h"
#include <iostream> #include<stdio.h> using namespace std; #define f(a,b) a##b
#define g(a) #a
#define h(a) g(a) #define A(x) T_##x
#define B(x) #@x
#define C(x) #x //如果EXP为真, 则输出宏参数“EXP”所指的字符串,
//注意:参数直接用真实参数替换。
#define WARN_IF(EXP) if(EXP) cerr << #EXP << endl;
//将红参数“n”,真实参数替换后
#define paster( n ) cout << "token" << #n << " = " << n << endl;
//两个宏参数,连接“宏参数”替换后形成字符串
#define _CONS(a, b) int(a##+##b)
#define _STRI(s) #s
#define STRI(s) _STRI(s) void test_sharp_symbol()
{
printf("%s\n", h(f(, ))); //输出:12 因为:h是转化的宏,其参数还是一个宏变量,则需要转化。
printf("%s\n", g(f(, ))); //输出:f(1,2) 因为:g直接把指代的参数以“字符串”输出
printf("%s\n", h(A())); // A(1)------〉T_1
printf("%d\n", B()); // B(1)------〉'1'
printf("%s\n", C()); // C(1)------〉"1" int div = ;
WARN_IF(div == ); //输出: div == 0
paster(); //输出: token9 = 9
cout << _CONS( + , ) << endl; //输出: 5 , 即: int(3+2), 输出5
cout << _STRI(INT_MAX) << endl; //输出: INT_MAX ,因为INT_MAX,本质上是一个宏,但是作为宏参数,则直接以字符串输出。
cout << STRI(INT_MAX) << endl; // prints : 2147483647 因为: INT_MAX ,再 STRI中已经替换为真实“数据”,从而可以以字符串输出
} int _tmain(int argc, _TCHAR* argv[])
{
test_sharp_symbol(); system("pause");
return ;
}

输出如下:

C/C++ 的宏中#和##的作用和展开的更多相关文章

  1. C/C++ 宏中的 #、#@、##的作用

    宏中的# 功能是将其后面的宏参数进行字符串化操作(Stringizing operator), 简单说就是在它引用的宏变量的左右各加上一个双引号. #define STRING(x) #x 下面二条语 ...

  2. C/C++语言中#的神奇作用:把宏参数字符串化/贴合宏参数

    宏中"#"和"##"的用法 一.一般用法   我们使用#把宏参数变为一个字符串,用##把两个宏参数贴合在一起. #define STR(s)      #s # ...

  3. C函数和宏中的可变参数

    一:调用惯例 函数的调用方和被调用方对函数如何调用应该有统一的理解,否则函数就无法正确调用.比如foo(int n, int m),调用方如果认为压栈顺序是m,n,而foo认为压栈顺序是n, m,那么 ...

  4. web.xml中load-on-startup的作用

    如下一段配置,熟悉DWR的再熟悉不过了:<servlet>   <servlet-name>dwr-invoker</servlet-name>   <ser ...

  5. C#中构造函数的作用

    C#中构造函数的作用 共同点: 都是实例化对象,初始化数据的 默认构造是说所有的类都从祖先object那继承了空参的构造方法,你不写与写空参构造都存在,而有参数的构造一般是自己写的,写就有不写就没有, ...

  6. MySQL数据库中delimiter的作用概述

    以下的文章主要是向大家描述的是MySQL数据库中delimiter的作用是什么?我们一般都认为这个命令和存储过程关系不大,到底是不是这样的呢?以下的文章将会给你相关的知识,望你会有所收获. 其实就是告 ...

  7. js中getBoundingClientRect的作用及兼容方案

    js中getBoundingClientRect的作用及兼容方案 1.getBoundingClientRect的作用 getBoundingClientRect用于获取某个html元素相对于视窗的位 ...

  8. Linq中关键字的作用及用法

    Linq中关键字的作用及用法 1.All:确定序列中的所有元素是否都满足条件.如果源序列中的每个元素都通过指定谓词中的测试,或者序列为空,则为 true:否则为 false. Demo: 此示例使用 ...

  9. JAVA中protected的作用

    JAVA中protected的作用   1.public:public表明该数据成员.成员函数是对所有用户开放的,所有用户都可以直接进行调用 2.private:private表示私有,私有的意思就是 ...

随机推荐

  1. Android 开发 创建WiFi、WiFi热点 ---开发集合

    WIFI 权限 <uses-permission android:name="android.permission.CHANGE_WIFI_STATE" /> < ...

  2. jquery之过滤filter,not

    <body> <h1>欢迎来到我的主页</h1> <p>我是唐老鸭</p> <p class="intro"> ...

  3. ISNUMERIC使用说明和BUG

    ISNUMERIC ( expression )参数 expression 要计算的表达式.返回类型 int 备注当输入表达式的计算结果为有效的 numeric 数据类型时,ISNUMERIC 返回 ...

  4. Halcom学习笔记1——Halcon知识点

    文件: 1.浏览HDevelop示例程序 2.程序另存在:Ctrl+Shift+S 3.导出:Ctrl+Shift+O X 编辑: 1.快捷键:  F3 激活     F4 注销     重复查找:C ...

  5. Quartz使用

    背景 很多时候,项目需要在不同时刻,执行一个或很多个不同的作业. Windows执行计划这时并不能很好的满足需求了,迫切需要一个更为强大,方便管理,集群部署的作业调度框架. 介绍 Quartz一个开源 ...

  6. 绑定checkedComboBox

    using System; namespace CommonLib{ /// <summary> /// CommonCode 的摘要说明. /// </summary> [S ...

  7. 结构体指offsetof宏详细解析

    1.#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE*)0)->MEMBER)     (include/linux/stddef.h) ...

  8. 582. Kill Process杀死所有子代

    [抄题]: Given n processes, each process has a unique PID (process id) and its PPID (parent process id) ...

  9. 拜托!面试请不要再问我Spring Cloud底层原理[z]

    [z]https://juejin.im/post/5be13b83f265da6116393fc7 拜托!面试请不要再问我Spring Cloud底层原理 欢迎关注微信公众号:石杉的架构笔记(id: ...

  10. 最短路径(SP)问题相关算法与模板

    相关概念: 有向图.无向图:有向图的边是双行道,无向图的边是单行道.在处理无向图时,可以把一条无向边看做方向相反的两条有向边. 圈 cycle / 回路 circuit:在相同顶点上开始并结束且长度大 ...