首页 > 编程语言 >C++随笔:内联(inline)

C++随笔:内联(inline)

时间:2023-05-16 18:11:24浏览次数:54  
标签:定义 代码 C++ bilibili 编译器 内联 inline 函数

内联(inline)

    内联函数以牺牲源文件大小为代价减小程序运行的时间,内联函数类似于一种编译器控制的复制/粘贴功能,当一个函数内部逻辑执行的消耗要远小于函数调用本身的消耗时,会额外增加栈空间的消耗,而被标记为内联的函数,编译器会直接将函数的定义“粘贴”到代码的调用处,直接在调用处将代码展开,这样就不需要执行函数调用了,但也增加了冗余的代码量。

    同时,内联也是宏的一种增强,宏替换是简单的文本替换,不会对类型进行检查,且有时候会带来逻辑上的错误或歧义,比如经典的乘法宏二义性定义:

#define MUL(x, y) x * y
int main()
{
// 按照正常逻辑,3+4的和,与5相乘,结果是35
// 但宏替换仅仅是简单的文本替换,实际执行的是:3 + 4 * 5,结果为23
cout << MUL(3 + 4, 5) << endl;
return 0;
}

// 踩坑之后,也可以通过规范化来避免宏定义的这种问题,总结就是:多用括号!
#define MUL(x, y) ((x) * (y))


内联增强了宏的功能:

① 内联可以打断点调试,宏不能打断点调试。(可以加log输出,宏定义可以替换为一个函数,宏定义本身不是函数)

② 内联会做类型和语法检查,宏不会。比如:

#define MACRO_PRINT(n) \
{ \
printf("%d\n", n); \
}

inline void InlinePrint(int n)
{
printf("%d\n", n);
}

int main()
{
// 会有参数不匹配的警告,但在可以的情况下,不会进行隐式转换
// 输出-1099511628,不同环境下不同,总之不是一个正常的数字
MACRO_PRINT(66.6666);
// 也会有参数不匹配的警告,但是double到int可以隐式转换
// 输出66,隐式转换导致数据丢失
InlinePrint(66.6666);
return 0;
}


需要特别注意的是:

① 与宏定义在预编译阶段处理不同,内联在编译阶段处理。

② 内联函数是一个函数,将宏的定义内容书写为一个函数,不代表这个宏就表示一个函数,它只是用来替换的一段文本。

③ 对编译器来说,内联只是建议,编译器会根据实际情况决定是否内联,比如编译器自身的设定,又比如判断函数内部代码量或执行逻辑的开销远大于调用开销等情况,编译器会自行判断是否接受程序员对函数的内联标记。有意思的是,编译器也可能会自行判断将一个没有被标记为内联的函数标记为内联。并且,内联只会在release版本生效,debug版本都当作普通函数处理,这里可以看出,内联本质上是一种编译器对代码的优化。

④ 内联针对的是函数,不是只有类的成员函数可以内联,普通定义的函数也可以内联,类中声明和定义的函数,会被编译器默认为内联(不过还是加上关键字,便于阅读)。

⑤ 内联的函数声明定义最好在同一个文件或模块中,内联必须对编译器可见,如果声明定义分开,不会有语法和编译上的问题,但连接时,因为找不到内联函数的定义,无法在调用处将代码展开,会报连接错误。当然,也可以将声明和定义的文件或模块都包含进来解决这个问题,但最好都放在同一个文件或模块中。

⑥ 在代码量较多(比如超过10行),或有循环,递归时,不要用内联,用了也不一定会被编译器内联。

 

 

B站个人主页:https://space.bilibili.com/513745196

知乎个人主页:https://www.zhihu.com/people/LingZhiZi

微信公众号搜索:灵知子 作者:灵知子 https://www.bilibili.com/read/cv13225561 出处:bilibili

 

 

https://www.bilibili.com/read/cv13225561

标签:定义,代码,C++,bilibili,编译器,内联,inline,函数
From: https://www.cnblogs.com/im18620660608/p/17406437.html

相关文章

  • C++
    计算时间相减定义一个时间类,小时和分钟是其两个私有成员数据。输入一个起始时间和一个结束时间(起始时间早于结束时间),通过运算符重载-(减号),计算这两个时间相隔多少分钟。说明:这两个时间在同一天之内,且采用24小时计时分式,即从00:00-23:59。输入格式:测试输入包含若干测试用例,每个......
  • C++:全局变量和static变量初始化
    (一)全局变量全局变量、文件域中的静态变量、类中的成员静态变量在main函数执行前初始化;局部变量中的静态变量在第一次调用时初始化。C++保证,全局变量在main函数第一次使用它之前,就把它初始化好。(可以直接理解成在main函数执行前就初始化好)但这个初始化可细分为:编译时初始化和运......
  • c++输入输出流对象
    实验项目名称:输入输出流一、实验目的掌握文本文件和二进制文件的基本访问方法;了解一般I/O流和文件流的关系;了解文件与文件流的关系;了解文件系统的概念,包括文件指针和关于文件的操作;掌握文件类的定义和相关操作的定义、使用方法;掌握利用常用函数进行文件的打开、关闭、读写......
  • c++打卡练习(31)
    求两个整数的最大公因数流程图:伪代码:源代码:#include<iostream>usingnamespacestd;intmain(){ intx,y,i,k; cout<<"请输入两个整数"<<endl; cin>>x; cin>>y; if(x<y){ intt; t=x; x=y; y=t; } for(i=y;i>=1;i--){ if(x%i==0&&y%i=......
  • c++打卡练习(30)
    不重复的三位数用1234这四个数组成不重复的三位数,每个数位的数也不同,问都有谁流程图:伪代码:源代码:#include<iostream>usingnamespacestd;intmain(){ inta,b,c,d; for(a=1;a<=4;a++){ for(b=1;b<=4;b++){ for(c=1;c<=4;c++){ if(a!=b&&a!=c&&b!=c){ cout<<a&......
  • C++ | 运算符重载
    1.运算符重载1.1基本概念函数重载(函数多态)是指用户能够定义多个名称相同但参数列表不同的函数,旨在使用户能够用同名的函数来完成相同的基本操作,即使这种操作被用于不同的数据类型。运算符重载将重载的概念扩展到运算符上,允许赋予C++运算符多种含义。实际上,很多C++(也包括C语......
  • 《c++徒步》vs界面详解
    vs2010字符集:多字节字符集和Unicode字符集是不同的字符集标准,主要区别如下:字符编码范围不同:多字节字符集使用单字节或多个字节来表示字符,编码范围较窄,通常只能表示本地语言的字符集。Unicode字符集则是全球通用的字符编码标准,可以同时表示全球范围内的多种语言字符集。字......
  • C++ 图进阶系列之剖析二分图的染色算法和匈牙利算法
    1.前言二分图又称作二部图或称为偶图,是图论中的一种特殊类型,有广泛的应用场景。什么是二分图?二分图一般指无向图。看待问题要有哲学思想,有二分图也可以是有向图。如果图中所有顶点集合能分成两个独立的子集,且任一子集中的任意顶点之间没有边连接,则称这样的图为二分图。......
  • 编写C++时遇到"undefined reference to `std::cout'"
    今天直接将C语言的配置文件粘贴到了C++的工程里面,代码没有报错但是编译报错后面自己试了一下发现用gcc编译c++的话就会报"undefinedreferenceto`std::cout'"的错误解决方法:用g++重新编译......
  • C/C++ CreateFileMapping 共享内存
    原链接C/C++CreateFileMapping共享内存-lyshark-博客园(cnblogs.com)#include<iostream>#include<Windows.h>usingnamespacestd;intmain(){HANDLEhMap;PINTlpMapAddr;//getfilemapping'shandlehMap=OpenF......