C++学习基础十六-- 函数学习笔记
C++ Primer 第七章-函数学习笔记
一步一个脚印、循序渐进的学习。
一、参数传递
- 每次调用函数时,都会重新创建函数所有的形参,此时所传递的实参将会初始化对应的形参。
- 「如果形参是非引用类型,则复制实参的值来初始化形参;如果形参是引用类型,则形参只是实参的别名。」
- 「非引用形参表示对实参的局部副本,函数内修改此类型形参时仅仅改变局部副本的值,一旦函数执行结束,这些局部变量的值就没有了,因此不影响实参的值。」
- 「如果函数参数为指针,同样形参是实参的副本,修改形参指针的值不影响实参,但是如果修改形参指针所指向的值则会影响实参。」
1 void reset1(int *p) 2 { 3 *p = 0;//修改形参所指向的值,则实参所指向的值也会改变,变为0 4 } 5 6 void reset2(int *p) 7 { 8 p = 0;//仅修改形参值,不影响实参值,即实参还是原来的地址,而形参的地址变为0 9 } 10 11 void main() 12 { 13 int i = 2; 14 reset1(&i); 15 printf("i = %d \n",i);// i = 0 16 17 i = 2; 18 reset2(&i); 19 printf("i = %d \n",i); // i = 2 20 }
- 「如果想要保护指针指向的值,则需要将指针定义为const,表示当前指针所指向的值可读不可写。」
1 void reset3(const int *p) 2 { 3 p = 0;//ok 4 *p = 0;//error,不允许修改p所指向的值 5 }
- 可以用非const指针初始化const指针,但不允许使用const指针初始化非const指针。例如:
int i = 2; const int *p1 = &i;//ok, 非const指针初始化const指针 int p2 = p1;//error, 不能用const指针初始化非const指针,报语法错误
- 当函数的参数是非引用(或非指针)非const的类型时,在调用该函数时,实参既可以是const类型的,也可以是非const类型的。
-
如果需要在函数内部修改实参值,则需要将形参定义为引用或者指针。
-
const修饰变量,表示该变量不可修改,这是对于基本数据类型而言。对于指针来说,有两种情况:
1 int num = 2; 2 int num2 = 30; 3 const int *p1 = #// *p1 = 20 不允许修改 4 int * const p2 = #// p2 = &num2 不允许
- 第三行这种写法表示p1所指向的值是const类型的,不允许修改其所指向的值。
- 第四行这种写法表示p2这个指针是const类型的,不允许给该指针重新赋值。
- 指向指针的引用
写法:int *&p1
从右向左理解,表示p1是一个引用,与该引用绑定的是一个int型的指针。
这种写法一般常用函数的参数,
举例说明:实现一个函数,该函数的功能是交换两个参数的值。
这有三种方法:
- 第一种方法:传递int型指针
这种情况下,形参复制实参的值,此时两者的地址值相同,即指向相同的对象。此时只是修改了指针所指向地址中的值,而并未改变指针值。
1 void TransNums(int *p1, int *p2) 2 { 3 int nTmp = (*p1); 4 *p1 = *p2; 5 *p2 = nTmp; 6 } 7 8 void main() 9 { 10 int num = 10; 11 int *p2 = # 12 int numP3 = 30; 13 int *p3 = &numP3; 14 15 printf("未交换-指针值:p2 = %d, p3 = %d num = %d numP3 = %d\n", *p2, *p3,num,numP3); 16 TransNums(p2, p3); 17 printf("交换-指针的值:p2 = %d, p3 = %d num = %d numP3 = %d\n", *p2, *p3,num,numP3); 18 }
打印结果:
未交换-指针值:p2 = 10, p3 = 30 num = 10 numP3 = 30
交换-指针的值:p2 = 30, p3 = 10 num = 30 numP3 = 10
「表示修改了指针所指向的值,而并未改变指针」
- 第二种方法:传递int性引用
形参是实参的别名,等同于实参。
1 void TransNumsRef(int &num1, int &num2){ 2 int nTmp = num1; 3 num1 = num2; 4 num2 = nTmp; 5 } 6 7 void main() 8 { 9 int num = 10; 10 int numP3 = 30; 11 12 printf("未交换-实参值:num = %d, numP3 = %d\n", num, numP3); 13 TransNumsRef(num, numP3); 14 printf("交换-实参的值:num = %d, numP3 = %d\n", num, numP3); 15 }
打印结果:
未交换-实参值:num = 10, numP3 = 30
交换-实参的值:num = 30, numP3 = 10
- 第三种方法:传递指针引用参数
此时形参为引用,指向指针,这种情况下修改了指针,即指针的值(或地址)改变了
1 void TransNumsPRef(int *&p1, int *&p2) 2 { 3 int *pTmp = p1; 4 p1 = p2; 5 p2 = pTmp; 6 } 7 8 void main() 9 { 10 int num = 10; 11 int *p2 = # 12 int numP3 = 30; 13 int *p3 = &numP3; 14 printf("未交换-指针引用值:p2 = %d, p3 = %d num = %d numP3 = %d\n", *p2, *p3,num,numP3); 15 TransNumsPRef(p2, p3); 16 printf("交换-指针引用的值:p2 = %d, p3 = %d num = %d numP3 = %d\n", *p2, *p3,num,numP3); 17 }
打印结果:
未交换-指针引用值:p2 = 10, p3 = 30 num = 10 numP3 = 30
交换-指针引用的值:p2 = 30, p3 = 10 num = 10 numP3 = 30
- 几种常见的表示形式:
1 int &arr[10]--- 表示arr是一个引用数组,即arr一个数组,数组的每个元素是int类型的引用 2 int (&arr)[10]--表示arr是一个数组的引用,数组的每个元素是int类型变量 3 int *arr[10]----表示arr是一个指针数组,即arr是一个数组,每个元素是一个int型指针 4 int (*arr)[10]--表示arr是一个数组的指针,即该指针指向一个数组,数组的每个元素是int变量,等同于int (arr*)[10]
二、函数声明
-
默认实参
声明一个函数时,我们可以给定形参的默认值,这种用法就是默认实参。如果有一个或多个形参具有默认实参,那么它后面的所有形参都必须有默认实参。
一般而言,具有默认实参的形参放在参数列表的后面。 -
应该在函数声明中提供默认实参。如果在函数定义的形参列表中提供默认实参,那么只有在包含该函数定义的源文件中调用,其默认实参才有效。
三、内联函数
- 优点:
- 减少函数调用的开销。将函数定义为内联函数,就是在程序中每个调用点上“内联的”展开。
- 注意事项:
- 不同于其他函数,内联函数定义必须在头文件中实现。
四、重载函数
-
重载函数:函数名相同,参数列表不同的函数。
-
函数名与参数列表完全相同,返回值类型不同的不能成为函数重载。即函数重载不能依赖于返回值类型。
-
重载和const参数:仅当形参是引用或指针时,形参是否为const才有影响。
-
几种情况并举例说明:
- 基于函数的引用形参是否为const实现函数重载
1 void lookat(int &a); 2 void lookat(const int &a);//新函数,即函数重载
「如果形参为非const引用,则函数调用时,不能将const类型的实参传递过来;如果传递了const对象,则必须调用带有const形参的函数。」
「如果形参为const引用,则函数调用时,既可以传递const对象,也可以传递非const对象。」
- 基于函数的指针形参是否指向const对象实现函数重载
1 void func(int *p); 2 void func(const int *p);//新函数,指向const对象
「如果形参为指向const对象的指针,则函数调用时,实参既可以是指向const对象的指针,也可以是指向非const对象的指针。」
「如果两个函数仅在指针形参是否指向const对象不同,则指向非const对象的指针形参对于指向非const对象的实参来说是最佳匹配。」
- 不能基于指针本身是否是const类型重载函数
1 void foo(int *p); 2 void foo(int * const p);//重复声明