1、 函数的参数
2、函数的嵌套调用
3、函数的递归调用
函数直接或间接的调用自身。
递推+回归 ==》递归
4 、引用
- 引用(&)是标识符的别名
- 定义一个引用时,必须同时对他进行初始化,使它指向一个已存在的对象
例如:
int i,j;
int &ri=i;//定义int引用ri,并初始化为变量i的引用
j=10;
ri=j;//相当于i=j - 一旦一个引用被初始化后,就不能改为指向其他对象
-
引用可以作为形参
引用做形参,可以实现双向传递。
单向传递
引用 实现双向传递
-
返回引用(一般被用在单目运算符的重载)
一般用作类对象的返回值,可以减少临时对象的申请、构造等操作,节省空间时间,比如
A& operator ++(){
//若干代码
return *this;//返回自身引用
}
为什么需要返回对象的引用?
——返回引用而不是返回对象,这将避免使用复制构造函数,从而提高效率
返回对象会涉及到生成返回对象的副本。因此,返回对象的时间成本包括了调用复制构造函数来生成副本所需的时间和调用析构函数删除副本所需的时间。返回引用可以节省时间和内存。直接返回对象与按值传递对象类似,他们都生成临时副本。同样,返回引用与按引用传递对象类似,调用和被调用的函数对同一个对象进行操作。
并不是总是可以返回引用的。比如函数不能返回在函数中创建的临时对象的引用。因为当函数结束调用时,临时对象将消失,因此这种引用是非法的。在这种情况下,应返回对象,以生成一个调用程序可以使用的副本。
5含有可变参数的函数
C++标准中提供了两种主要的方法
- 如果所有的实参类型都相同,可以传递一个名为initializer_list的标准库类型
- 如果实参的类型不同,我们可以编写可变参数的模板
initializer_list
- 定义:initializer_list是一种标准库类型,用于表示某种特定类型的值的数组,该类型定义在同名的头文件中
- 使用方法:initializer_list是一个类模板,在使用模板时,我们需要在模板名字后面跟一对尖括号,括号内给出类型参数
例如:
initializer_list<string> ls;//initializer_list的元素类型是string
initializer_list<int> li;//initializer_list的元素类型是in
- initializer_list比较特殊的一点是,其对象中的元素永远是常量值,我们无法改变initializer_list对象中元素值,含有initializer_list形参的函数也可以同时拥有其他形参
- 使用举例:在编写代码输出程序产生的错误信息时,最好使用统一的一个函数来实现该函数的功能,使得对所有的错误处理都能够整齐划一,然而错误信息的种类不同,调用错误信息输出函数时传递的参数也不同。因此我们可以用initializer_list编写一个错误信息输出函数使其作用于可变数量的形参。
6、内联函数
- 内联函数作用:让我们在调用简单函数时,提高效率。
- 内联函数的实现是靠编译器来实现的
- 内联函数的声明需要要到关键字:inline
内联函数:是C++中的一种特殊函数,它可以像普通函数一样被调用,在调用时并不通过函数调用的机制而是通过将函数体直接插入调用处来实现的,即编译时在调用处用函数体进行替换,节省了参数传递,控制转移等开销
注意:
- 内联函数体内不能有循环语句和swich语句
- 内联函数的定义必须出现在内联函数第一次被调用之前
- 对内联函数不能进行异常接口声明
- inline只是对编译器的一个建议,编译器对自己进行判断该函数是否能当做内联函数
使用:
inline 返回值类型 函数名(函数参数){
//此处定义函数体
}
7、constexpr (可以这样记const expr)常量
constexpr是C++11的新标准
- constexpr可以用来初始化常量
- constexpr限定在了编译期常量
- C++11中的constexpr指定的函数返回值和参数必须要保证是字面值,而且必须有且只有一行return代码
- 而C++14中只要保证返回值和参数是字面值就行了,函数体中可以加入更多的语句,方便了更灵活的计算。
constexpr函数语言规定 - constexpr修饰的函数,在其所有参数都是constexpr时,一定返回constexpr!
- const并不能代表“常量”,它仅仅是对变量的一个修饰,告诉编译器这个变量只能被初始化,且不能被直接修改(实际上可以通过堆栈溢出等方式修改)。而这个变量的值,可以在运行时也可以在编译时指定。
- constexpr可以用来修饰变量、函数、构造函数。一旦以上任何元素被constexpr修饰,那么等于说是告诉编译器 “请大胆地将我看成编译时就能得出常量值的表达式去优化我”。
const int func() {
return 10;
}
main(){
int arr[func()];
}
//error : 函数调用在常量表达式中必须具有常量值
constexpr func() {
return 10;
}
main(){
int arr[func()];
}
//编译通过
字面值
int a=5;//a是变量,5是字面值,即常量
8、带默认参数值的函数
带默认值的的参数:可以预先设定默认的参数值,调用时如给出实参,则采用实参值,否则采用预先设置的默认参数值。
例:
//预先设定了参数x的默认值为5,y为6
int add(int x=5,int y=6){
return x+y;
}
int main(){
add(10,20);//10+20
add(10);//10+6
add();//5+6
}
默认参数值的说明次序
- 有默认参数的形参必须列在形参列表的最后,即默认参数值的右边不能有无默认值的形参
- 调用时实参与形参的结合次序是从左到右
例:
int add(int x,int y=5,int z=6)//正确add(3)是3+5+6
int add(int x=5,int y,int z=6)//错误
int add(int x=5,int y=6,int z)//错误
默认参数值与函数的调用位置
- 不能在函数的声明和定义中都给出默认参数值
- 当只有函数的定义,或函数定义在前调用在后,则默认参数值可以在函数定义中给出
-
如果一个函数有原型声明,且原型声明在定义之前,则默认参数值在函数原型声明中给出
例:
//原型声明在前
int add(int x=5;int y=6);
int main(){
add();
}
int add(int x,int y){
//此处不能再指定默认值
return x+y;
}
//定义在调用之前
int add(int x=5,int y=6){
return x+y;
}
int main(){
add();
}
函数调用具体实例
9、函数的重载
函数的重载是C++多态性的一个重要机制,是由静态多态性实现的(编译时实现)
函数重载的语法规则:
形参的类型不同
int add(int x,int y);
float add(float x,float y);
形参的个数不同
int add(int x,int y);
int add(int x,int y,int z);
注意事项:
- 重载函数的形参必须不同:个数不同或类型不同
- 函数的返回值和形参名不同,不能用来区分重载函数
-
编译程序将根据实参和形参的类型及个数的最佳匹配来选择调用哪一个函数
- 不要将不同功能 的函数声明为重载函数,以免出现调用结果的误解、混淆。这样不好:
例子
10、C++系统函数
函数是表现功能模块的一种形式,很多常用的模块都在C++标准库实现好了,直接调用就可以。
- C++的系统库中提供了几百个函数可供程序员使用,例如:
- 求平方根函数sprt
- 求绝对值函数 abs
- 使用系统函数时要包含相应的头文件,例如:
- cmath//数学函数库