文章目录
- 函数重载
- 为什么C++支持重载,C语言不支持呢?
- extern “C”
- 引用再探
- 引用的特性
- 引用的使用场景
- 引用和指针
- 引用和指针的不同点:
- 内联函数
- 什么是内联函数?
- 内联函数的特性
- 内联函数的好处
- 类的内联成员函数的声明
- 内联函数的使用
- constexpr函数
- 概念
- 特征
- 内联函数和constexpr函数放在头文件内
函数重载
在同一个作用域下,对于相同的函数名,函数的参数类型不同,参数顺序不同,参数的个数不同, 都可以形成函数的重载(参数名不同,返回值不同不形成重载)
函数的重载主要用于处理功能相同,形参类型不同的数据。
void test(int i, int j)
{cout << "test" << endl;
}void test(double i, int j) // 类型不同
{cout << "test" << endl;
}void test(int j, double i) // 顺序不同
{cout << "test" << endl;
}void test(double i, int j, int k) // 个数不同
{cout << "test" << endl;
}
为什么C++支持重载,C语言不支持呢?
因为windows对函数重载的处理更加复杂,所以这里用linux下的gcc和g++来看更加直观。
首先我们要知道,链接器看到有函数被调用的时候,就会到符号表中去查找对应的函数名,来获取函数的地址,再链接到一起
先看C语言是怎么处理的
通过反汇编我们可以看到,C语言并没有对函数名进行处理,也就是说无论我们参数的个数,参数的类型,参数的顺序怎么修改,它只认函数名,如果出现了第二个相同函数名的,就算重定义。
下面再看C++的:
这里可以看到,C++对函数名进行了处理,函数以_Z4开头,接着是函数名,最后是所有参数的缩写。
_Z是所有函数的前缀,4是函数名的字符个数,例如第一个_Z4testii则代表函数名为test,具有四个字符,参数类型缩写分别是ii。
这也就是为什么返回值不同和参数名不构成重载的原因,它们不被作为对函数特征的处理。C++正是通过这种函数名修饰规则来实现函数的重载。
extern “C”
有时候我们在使用C++的时候,对于某些函数,想让它按照C的风格来编译,那么就在函数前加extern “C”,意思是告诉编译器,将该函数按照C语言规则来编译。
引用再探
引用的特性
- 引用在定义的时候必须初始化(因为引用是某个对象的别名,所以必须初始化)
- 一个对象可以有多个引用
- 一旦引用一个实体,就不能再引用别的实体(有点类似指针的顶层const)
引用的使用场景
- 作为参数
struct A
{int arr[1000000];
};void test(A& s1)
{}
假设我们存在一个超级大的结构体,如果我们直接将结构体传过去的话,会产生一个临时变量来将这个结构体拷贝到形参中,这是极大的开销,但如果我们使用引用的话,传的只是一个别名而已,所有的操作还是在结构体本身上进行的,但是需要注意的和上面一样,如果我们要传递一个常量,就必须要在引用前加上const。
struct A
{int arr[1000000];
};void test (const A& s1)
{}
int main(int argc, char const *argv[])
{const A a = {10,324,32};test(a);return 0;
}
- 作为返回值
int& Add(int a, int b)
{int c = a + b;return c;
}int main()
{int& ret = Add(1, 2);Add(3, 4);cout << ret << endl;return 0;
}
对于这样一个代码,我们可能第一眼觉得ret会是3。
但是其实是7。
因为我们返回的是c的一个引用,但是c只存在于调用时的那个栈帧,调用结束后那个栈帧就会被销毁,虽然销毁后数据不会被清空,但是那片区域的访问权限就会被放开,有可能会被下次调用的函数使用,也有可能会被其他的一个操作给使用,所以这是一种极为不安全的行为。
上面的7是第二次调用后修改了c的值。
所以,如果需要引用作为返回值,就必须保证出了函数作用域,返回的对象没有归还给系统,仍然存在。
以值作为参数或者返回值时,在传参和返回的时候,都会传递或返回原变量的一个临时的拷贝,这样的效率是非常低下的,尤其是数据特别大的时候,但如果使用引用作为参数的话,就不会有这样的问题。
引用和指针
语法概念上:引用是对象的一个别名,没有独立的空间,和其引用的实体共用一个空间。
但我们发现,引用其实和指针很像,它更像一个顶层const的指针,所以我们可以进入反汇编看看他们之间有没有关系
int main()
{int x = 5;int& y = x;int* z = &x;return 0;
}
反汇编下我们可以看到,指针和引用在汇编下的实现是一模一样的。
所以我们可以得出一个结论:引用是按照指针来实现的,在指针的基础上又给他封装了新的功能。
引用和指针的不同点:
- 引用在定义时必须初始化,指针没有要求
- 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型 实体
- 没有NULL引用,但有NULL指针
- 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占 4个字节)
- 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小
- 有多级指针,但是没有多级引用
- 访问实体方式不同,指针需要显式解引用,引用编译器自己处理
- 引用比指针使用起来相对更安全
内联函数
什么是内联函数?
用inline关键字修饰的函数就是内联函数,在编译时编译器会将函数的代码在调用内联函数的地方展开,减去了函数压栈的开销,提升程序运行的效率(牺牲空间换取时间)。
例如这样一个简单的代码
如果我们直接调用它
在汇编下可以看到,他会创建一个新的栈帧,将参数3,4压栈,然后计算完再返回结果
而如果在函数前面加上inline使其变为内联函数
这时再看,就会发现它直接把函数的代码在调用处直接展开,不会再创建新的栈帧。
内联函数的特性
- 内联函数是一种用空间换时间的做法,省去了创建栈帧和压栈的开销,但也因此代码很复杂和具有循环或递归之类的函数不适合作为内联函数,就算声明为内联函数编译器也会自动将其忽略。
- 内联函数不能声明和定义分离,因为一旦声明为内联函数,在调用的时候就会直接展开,没有了函数的地址,就无法将其链接到定义的部分。
值得一提的是,内联函数与C语言中的宏函数有些类似,虽然宏的性能不错,但是因为宏缺乏类型的安全检查和无法调试(在预处理阶段就进行了宏替换),在C++中宏函数被内联函数替代,宏常量定义被const取代。
内联函数的好处
- 较之等价的表达式更易于阅读
- 可以被其他应用重复利用,省去了重新编写的代价
- 如需修改计算过程,显然修改函数比先找到等价表达式所有出现的地方再逐一修改更容易。
类的内联成员函数的声明
我们可以在类内把 inline
作为声明的一部分显式地声明成员函数,同样的,也能在类的外部用 inline
关键字修饰函数的定义(当然在声明和定义的地方同时说明 inline
也是合法,只是没有必要)。
内联函数的使用
- 滥用内联将导致程序变得更慢;
- 最好不要内联超过
10
行的函数; - 谨慎对待析构函数,析构函数往往比其表面看起来要更长,因为有隐含的成员和基类析构函数被调用;
- 内联那些包含循环或
switch
语句的函数常常是得不偿失 (除非在大多数情况下,这些循环或switch
语句从不被执行); - 有些函数即使声明为内联的也不一定会被编译器内联:比如虚函数和递归函数就不会被正常内联。
- 通常,递归函数不应该声明成内联函数。(递归调用堆栈的展开并不像循环那么简单, 比如递归层数在编译时可能是未知的,大多数编译器都不支持内联递归函数)。
- 虚函数内联的主要原因则是想把它的函数体放在类定义内,为了图个方便,亦或是当作文档描述其行为,比如精短的存取函数。
constexpr函数
概念
能用于常量表达式的函数
特征
- 函数的返回类型及所有形参的类型都得是字面值类型
- 函数体中必须有且只有一条return语句
- 编译器把对constexpr函数的调用替换成其结果值(constexpr函数被隐式地指定为内联函数)
- 函数体内允许包含 运行时不执行任何操作的语句
- 允许返回一个非常量,应用时编译器会进行检查。(constexpr不一定返回常量表达式)
内联函数和constexpr函数放在头文件内
和其他函数不同,内联函数和constexpr可以在程序中多次定义(每一次展开就是一次定义)。但多个定义必须完全一致,基于这个原因,内联函数和constexpr函数通常定义在头文件。