C++:在代码中理解宏定义和内联函数

宏定义

在C语言中,宏定义是比较常用的预处理指令,即使用“标识符”来表示“替换列表”中的内容。标识符称为宏名,在预处理过程中,预处理器会把源程序中所有宏名,替换成宏定义中替换列表中的内容。
常见的宏定义主要有两种,即不太参数的宏定义和带参数的宏定义(宏函数)。

不带参数的宏定义

格式:

#define 表示符 替换列表
/*替换列表可以是数值常量、
字符常量、字符串常量等,所
以可以把故可以把宏定义理解
为使用标识符表示一常量,或
称符号常量。
*/

注意

1.#可以不在首行,但是在它前面只允许有空格符

#define PI 3.12
 #define pi 3.2

int main()
{
	return 0;
}

在这里插入图片描述

int a; #define pi 3.2

int main()
{
	return 0;
}

在这里插入图片描述

2.标识符和替换列表之间不能用赋值符=,而且替换列表后不能加分号

#define PI = 3.12
//int a;#define P 3.14
#define pi 3.2;

int main()
{
	return 0;
}

在这里插入图片描述
这个地方虽然语法没有错误,但是当我们使用的时候就会出现错误了;

#include <stdio.h>
#include <Windows.h>
//#define PI =3.12
//int a;#define P 3.14
#define pi 3;
//在double b = 2 * pi;这条语句可以通过编译,
//但是在int a[pi];就不可以,因为pi被替换成了3;

int main()
{
	//double a = 2 * PI;
	double b = 2 * pi;
	printf("%lf", b);
	int a[pi];
	system("pause");
	return 0;
}

在这里插入图片描述

3.由于宏定义仅是做简单的文本替换,故替换列表中如有表达式,必须把该表达式用括号括起来,否则可能会出现逻辑上的“错误”。

#include <stdio.h>
#define N 3+3
int main()
{
	int a = N * N;
	printf("%d\n", a);
	system("pause");
	return 0;
}

在这里插入图片描述
我们给他加上括号之后

#include <stdio.h>
#define N (3+3)
int main()
{
	int a = N * N;
	printf("%d\n", a);
	system("pause");
	return 0;
}

在这里插入图片描述

4.当替换列表一行写不下时,可以使用反斜线\作为续行符延续到下一行。

#include <stdio.h>
#define ch "中华\
人民共\
和国是……"
int main()
{
	printf("%s\n", ch);
	system("pause");
	return 0;
}

在这里插入图片描述

带参数的宏定义(宏函数)

格式:

#define 标识符(参数1,参数2,...,参数n) 替换列表
#include <stdio.h>

#define MAX(a,b) (a > b ? a:b)
int main()
{
	int a = 10, b = 20;
	printf("%d\n", MAX(a,b));
	system("pause");
	return 0;
}

在这里插入图片描述
注意

  1. 标识符与参数表的左括号之间不能有空格,否则预处理器会把该宏理解为普通的无参宏定义,故以下是错误的带参宏定义形式
    在这里插入图片描述
    在这里插入图片描述
    2.宏替换列表中每个参数及整个替换列表,都必须用一对小括号 () 括起来,否则可能会出现歧义
#include <stdio.h>
#define MUL(a,b) (a*b)
int main()
{
	int c;
	c = MUL(3, 5 + 1);
	printf("c=%d\n", c);
	system("pause");
	return 0;
}

在这里插入图片描述
这里我们只要把可能发生歧义的地方加上括号就可以了

#include <stdio.h>
#define MUL(a,b) ((a)*(b))
int main()
{
	int c;
	c = MUL(3, (5 + 1));
	printf("c=%d\n", c);
	system("pause");
	return 0;
}

在这里插入图片描述
但是大家看下面的这个程序,它运行的结果是22,这是为什么呢?

#include <iostream>
using namespace std;

#define MAX(a,b) (a > b ? a:b)
int main()
{
	int a = 10;
	int b = 20;
	cout << MAX(a, ++b) << endl;
	system("pause");
	return 0;
}

在这里插入图片描述
其实就是在宏函数中所有的b都被替换成了++b,所以b就被加了两次,但是这显然不是我们想要的结果。这就涉及到宏函数的缺陷了。

宏函数的优缺点

优点

1.提高了程序的可读性,方便修改
2.提高程序的运行效率:使用带参的宏定义既可以完成函数调用的功能,又能避免函数的出栈与入栈操作,减少系统的开销,提高运行效率
3.宏是预处理器处理的,通过字符操作可以完成很多编译器无法实现的功能,如##连接符。

缺陷

1.代码可能相对较多
2.嵌套定义过多可能会影响程序的可读性,而且容易出错
3.对于带参数的宏而言,由于是直接替换,并不会检查参数是否合法,存在安全隐患。

内联函数

在C++中,引入了内联函数来解决宏函数的缺陷。

内联函数:以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数压栈的开销,内联函数提升程序运行的效率。
接下来我们通过反汇编来对比一下内联函数和普通函数的区别:
普通函数:

int Add(int a, int b)
{
	return a + b;
}
int main()
{
	int ret = Add(1, 2);
	cout << ret << endl;
	system("pause");
	return 0;
}

在这里插入图片描述
内联函数:

inline int Add(int a, int b)
{
	return a + b;
}
int main()
{
	int ret = Add(1, 2);
	cout << ret << endl;
	system("pause");
	return 0;
}

在debug模式下查看,需要对编译器进行设置,否则不会展开(因为debug模式下,编译器默认不会对代码进
行优化,以下给出vs2013的设置方式)
设置之前
在这里插入图片描述
设置方式

在这里插入图片描述

在这里插入图片描述
设置之后
在这里插入图片描述
在release模式下,查看编译器生成的汇编代码中不存在call Add
在这里插入图片描述

内联函数的特性

1.inline是一种以空间换时间的做法,省去调用函数额开销。所以代码很长或者有循环/递归的函数不适宜使用作为内联函数。
2. inline对于编译器而言只是一个建议,编译器会自动优化,如果定义为inline的函数体内有循环/递归等等,编译器优化时会忽略掉内联。
3. inline不建议声明和定义分离,分离会导致链接错误。因为inline被展开,就没有函数地址了,链接就会找不到。

  • 13
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
回答: C++的函数宏定义内联函数有一些区别。函数宏定义是在预编译阶段将宏名替换为宏体,它不是真正的函数,而是一种字符串替换的机制。宏定义没有类型检查,无论对还是错都会直接替换。而内联函数是真正的函数,使用inline修饰,编译器会在调用内联函数的地方展开,没有函数压栈的开销,提高了程序的运行效率。内联函数一般用于函数体的代码比较简单的函数,不能包含复杂的控制语句,如while、switch,并且内联函数本身不能直接调用自身。如果内联函数的函数体过大,编译器会自动将其变为普通函数。内联函数在编译阶段进行代码插入,而宏定义是在预编译阶段进行代码展开。因此,内联函数在编译时会进行类型检查,而宏定义没有类型检查。此外,内联函数的使用可以提高程序的效率,而宏定义的使用可以省去函数调用的开销。\[1\]\[2\] 然而,需要注意的是,宏定义内联函数在使用时都是进行代码展开。对于宏定义,预处理器会将所有的宏名替换为宏体;而对于内联函数,编译器会在每处调用内联函数的地方直接将内联函数的内容展开。这样可以省去函数压栈退栈的开销,提高了程序的效率。\[2\] 举个例子来说明宏定义内联函数的区别。假设有以下代码: #include <iostream> using namespace std; #define MAX(a,b) (a > b ? a:b) int main() { int a = 10; int b = 20; cout << MAX(a, ++b) << endl; return 0; } 这段代码宏定义MAX(a, b)会将a和b进行比较,并返回较大的值。在调用MAX(a, ++b)时,宏定义会将其展开为(a > ++b ? a:b)。由于宏定义是字符串替换,所以++b会被展开两次,导致b的值增加了两次。因此,输出结果为22。\[3\] #### 引用[.reference_title] - *1* *3* [C++:在代码理解宏定义内联函数](https://blog.csdn.net/qq_43825377/article/details/116211242)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [C++内联函数宏定义](https://blog.csdn.net/luolaihua2018/article/details/115377860)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值