title: 宏
date: 2021-06-21 14:52:01
tags: [C++]


宏中包含特殊符号

分为几种:###\

字符串化操作符(#)

在一个宏中的参数前面使用一个#,预处理器会把这个参数转换为一个字符数组,换言之就是:#是字符串化的意思,出现在宏定义中的#是把跟在后面的参数转换成一个字符串

注意:其只能用于有传入参数的宏定义中,且必须置于宏定义体中的参数名前。

例如:

#define exp(s) printf("test s is:%s\n",s)
#define exp1(s) printf("test s is:%s\n",#s)
#define exp2(s) #s 
int main() {
    exp("hello"); // test s is:hello
    exp1(hello);  // test s is:hello

    string str = exp2(bac);
    cout << str << " " << str.size() << endl; // bac 3
    
    // 忽略传入参数名前面和后面的空格。
    string str1 = exp2(   asda   bac   );
    /*
    当传入参数名间存在空格时,编译器将会自动连接各个子字符串,
    用每个子字符串之间以一个空格连接,忽略剩余空格。
    */
    cout << str1 << " " << str1.size() << endl; // asda bac 8

    return 0;
}

上述代码给出了基本的使用与空格处理规则,空格处理规则如下:

  • 忽略传入参数名前面和后面的空格。
#define exp(s) printf("test s is:%s\n",s)
#define exp1(s) printf("test s is:%s\n",#s)
#define exp2(s) #s 
int main() {
    string str = exp2(   bac  );
    cout << str << " " << str.size() << endl; // bac 3
    return 0;
}

输出:

bac 3
  • 当传入参数名间存在空格时,编译器将会自动连接各个子字符串,用每个子字符串之间以一个空格连接,忽略剩余空格。
#define exp(s) printf("test s is:%s\n",s)
#define exp1(s) printf("test s is:%s\n",#s)
#define exp2(s) #s 
int main() {
    string str1 = exp2(   asda  bac   );
    cout << str1 << " " << str1.size() << endl; // asda bac 8
    return 0;
}

输出:

asda bac 8

符号连接操作符(##)

##是一种分隔连接方式,它的作用是先分隔,然后进行强制连接。将宏定义的多个形参转换成一个实际参数名。

注意事项:

  • 当用##连接形参时,##前后的空格可有可无。
  • 连接后的实际参数名,必须为实际存在的参数名或是编译器已知的宏定义。
  • 如果##后的参数本身也是一个宏的话,##会阻止这个宏的展开。
示例:
#define expA(s) printf("前缀加上后的字符串为:%s\n",gc_##s)  //gc_s必须存在
#define expB(s) printf("前缀加上后的字符串为:%s\n",gc_  ##  s)  //gc_s必须存在
#define gc_hello1 "I am gc_hello1"
const char* gc_hello = "I am gc_hello";
int main() {
    expA(hello);    // I am gc_hello
    expB(hello1);   // I am gc_hello1
    return 0;
}

续行操作符(\)

当定义的宏不能用一行表达完整时,可以用\表示下一行继续此宏的定义。

注意\前留空格。

#define MAX(a,b) ((a)>(b) ? (a) \
   :(b))  
int main() {
    int max_val = MAX(3, 6);
    cout << max_val << endl;	// 6 
    return 0;
}

do{…}while(0)的使用

避免语义曲解

例如:

#define fun() f1();f2();
if (a > 0)
	fun()

这个宏被展开后就是:

if (a > 0)
	f1();
f2();

本意是 a > 0 执行 f1(),f2(),而实际是 f2() 不会执行,所以就错误了。

为了解决这种问题,在写代码的时候,通常可以采用{}块。如:

#define fun() {f1();f2();}
if (a > 0)
    fun();
// 宏展开
if (a > 0)
{
    f1();
    f2();
};

但是会发现上述宏展开后多了一个分号,实际语法不太对。(虽然编译运行没问题,正常没分号)

避免使用 goto 控制流

在一些函数中,我们可能需要在 return 语句之前做一些清理工作,比如释放在函数开始处由 malloc 申请的内存空间,使用 goto 总是一种简单的方法:

int f() {
    int* p = (int*)malloc(sizeof(int));
    *p = 10;
    cout << *p << endl;
#ifndef DEBUG
    int error = 1;
#endif
    if (error)
        goto END;
    // dosomething
END:
    cout << "free" << endl;
    free(p);
    return 0;
}

但由于 goto 不符合软件工程的结构化,而且有可能使得代码难懂,所以很多人都不倡导使用,这个时候我们可以使用 do{…}while(0) 来做同样的事情:

int ff() {
    int* p = (int*)malloc(sizeof(int));
    *p = 10;
    cout << *p << endl;
    do {
#ifndef DEBUG
        int error = 1;
#endif
        if (error)
            break;
        //dosomething
    } while (0);
    cout << "free" << endl;
    free(p);
    return 0;
}

这里将函数主体部分使用 do{…}while(0) 包含起来,使用 break 来代替 goto,后续的清理工作在 while 之后,现在既能达到同样的效果,而且代码的可读性、可维护性都要比上面的 goto 代码好的多了。

避免由宏引起的警告

内核中由于不同架构的限制,很多时候会用到空宏。在编译的时候,这些空宏会给出 warning,为了避免这样的 warning,我们可以使用 do{…}while(0) 来定义空宏:

#define EMPTYMICRO do{}while(0)

定义单一的函数块来完成复杂的操作

如果你有一个复杂的函数,变量很多,而且你不想要增加新的函数,可以使用 do{…}while(0),将你的代码写在里面,里面可以定义变量而不用考虑变量名会同函数之前或者之后的重复。 这种情况应该是指一个变量多处使用(但每处的意义还不同),我们可以在每个 do{…}while(0) 中缩小作用域,比如:

int fc()
{
    int k1 = 10;
    cout << k1 << endl;
    do {
        int k1 = 100;
        cout << k1 << endl;
    } while (0);
    cout << k1 << endl;
}

PDF:链接 密码:431u

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值