<>【C++学习】——(八)函数
<>基础
函数:封装了一段代码,可以在一次执行过程中被反复调用,包含函数头和函数体;
函数头:
* 函数名称(标识符),用于后续的调用;
* 形式参数,代表函数的输入参数;
* 返回类型,函数执行完成后返回结果的类型;
函数体:一个语句块(block),包含具体的计算逻辑;
函数的声明与定义:
* 函数声明只包含函数头,不包含函数体,通常在头文件中;
* 函数声明可以出现多次,定义通常出现一次(也有例外);
函数调用:
* 需要提供函数名与实际参数;
* 实际参数拷贝初始化给形式参数;
* 返回值会拷贝给函数的调用者;
* 栈帧结构(可自行了解下);
<>参数
*
对于非模板函数来说,其每个形参都有确定的类型,但形参可以没有名称;
*
实参到形参的拷贝顺序是不确定的;
*
函数的形参的传递一般分为:传值、传址、传引用;
*
变长参数的定义:
1、使用initializer_list传递:
#include <initializer_list> void fun(std::initializer_list<int> a){} int main
{ fun({1, 2, 3, 4}) }
注意:该方法只能传递类型相同的变长参数;
2、可变长度模板参数
3、使用省略号表示形式参数(一般不使用)
*
函数的缺省实参注意点:
1、如果某个形参具有缺省参数,那么它右侧的形参都必须具有缺省实参;
void fun(int x=1, int y=2){} // 这里y必须给定缺省值
2、具有缺省实参的函数调用时,传入实参按照从左到右的顺序进行匹配;
3、在一个翻译单元中,每个形参的缺省实参只能定义一次;
4、缺省实参为对象时,实参的缺省值会随对象值的变化而变化;
*
main函数的版本
1、无形参版本(一般使用)
2、带形参版本
int main(int argc, char *argv[]) {}
argc是非负数,表述传入参数个数,argv是一个指针指向传输参数的数组头。
<>返回类型
*
返回类型的几种书写方式:
1、经典方法:位于函数头的前部,也是最常规的写法;
2、C++11引入的方式:位于函数头的尾部;
auto fun(int x) -> int { return x*2; }
3、C++14引入的方式:返回类型的自动推导;
auto fun(int a) { return a; // 会根据return语句进行推导 }
<>函数重载与解析
函数重载:使用相同的函数名定义多个函数,每个函数具有不同的参数列表;
注意:不能基于不同的返回类型进行重载;
名称查找:
* 分为限定查找和非限定查找:有无限定在某个作用域中;
* 非限定查找会进行域的逐级查找——名称隐藏;
* 查找通常只会在已声明的名称集合中进行;
重载解析:在名称查找的基础上进一步选择合适的调用函数;
* 过滤不能被调用的版本:参数个数不对、无法将实参转为形参、实参不满足形参的限制条件;
<>内联函数
定义:将比较简单的函数逻辑展开到调用函数的部分,避免栈帧销毁,提升性能;
关键字:inline,如果一个函数在多个翻译单元展开,加入这个关键字可以避免重复定义;
<>constexpr函数
定义:之前有介绍常量表达式时用到了该关键字,现在对于函数也可以用该关键字;
作用:使得函数在编译器被执行,当然在有变量情况下也可在运行期执行;
constexpr int fun(int x){ // int y; std::cin >> y; 会报错,该语句需要用户传入参数,只能在运行期执行
return x * 2; } int main { constexpr int x = fun(2); // 编译器会翻译成 move eax 4,
去掉constexpr也可以 return x; }
注意:constexpr函数中的语句必须是可以在编译器执行的语句;
拓展:关键字consteval(C++20引入),函数只能在编译器执行;
<>函数指针
作用:可以用于高阶函数中,将函数指针作为参数;
代码案例:
int add(x) { return x + 1}; using T = int(int); int fun(K* F, int x) { int tmp
= (*F)(x); return tmp * 2; } int main { std::cout << fun(&add, 50) <<
std::endl; }
说明:这就是用函数指针定义的一个高阶函数,在之后的很多高阶函数、泛型算法中也是这样的用法;
注意:当函数对象进行赋值或者返回值时,返回的是一个函数指针类型的对象;
<>思考
1、我们常常会见到如下代码,是由什么作用?
extern "C" int fun(int x, int y) { return x + y; }
C语言对于函数是不能重载的,当用C调用C++程序时,往往找不到C++编译后的函数名,可通过如上代码定义一个函数为C类型函数;
2、可以用别名定义一个函数类型吗?
using X = int[3]; X a; // 这是定义了一个数组,同int a[3] using X = int(int); X fun; //
这是定义了一个int返回类型的函数
函数也是有类型的,可以用别名定义,并且函数类型不包含形参名称,并且只能声明,不能定义;
<>总结
本篇主要介绍了函数的基础概念以及一些特殊的函数方法和类型。重点需要注意的就是函数重载以及函数指针,这个在后续的模板以及泛型编程都会用到;下一篇将讨论
C++中的内存,也是最重要的一个部分。