C++学习基础五之函数参数——形参
一、理论部分
C++中函数形参主要分为两类,如图1所示,
图1
总结:
一、当函数参数为非引用形参时,传进函数体内的是实参的拷贝,(注意,对于基本类型而言,拷贝的是实参的值,对于指针而言拷贝的是实参的地址)
(1)若形参为非const的基本类型,则即可接收const实参,也可接收非const实参。只是在函数体内修改形参的值不影响实参的值。
因为对于基本类型的形参而言,传递进函数体的是实参拷贝的值,而不是实参本身,所以在函数体内修改实参的值不影响实参。
(2)若形参为非const的指针类型,则即可接收const实参,也可接收非const实参。在函数体内修改形参的值会影响实参的值。
因为对于指针类型的形参而言,传递近函数体的是实参指针地址的拷贝,形参指针和实参指针指向同一个地址,修改实参指针所指向的值会影响实参指针的所指向的值。
(3)若形参为const的基本类型,则即可接收const实参,也可接收非const实参。只是在函数体内不能修改形参的值。
(4)若形参为const的指针类型,则只能接收const类型的指针实参,不能接收非const的指针实参。
const指针的初始化规则:可以将指向const对象的指针指向非const对象,不可以将指向为非const对象的指针指向const对象。
二、当函数为引用形参时,传递近函数体内的是实参本身,因为引用就是变量的别名,改变引用的值同时会影响变量的值。
(1)引用形参是实参的别名,可以达到修改实参的目的。
(2)引用形参可以额外的返回值。
(3)对于函数体内不修改形参值的情况,应该将形参设置为const &,即const引用,可减少值的拷贝赋值。
(4)指向指针的引用若在函数体进行改变,则改变的是指针所指向对象的值。
二、代码片段如下:
代码片段1:
int add_1(int x,int y) { return x+y; } int add_2(const int x,const int y) { //x++;//error,const参数不能修改 return x+y; } int main() { int a=1,b=2; const int m=9; const int n=8; add_1(a,b);//ok add_1(m,n);//ok add_2(m,n);//ok add_2(a,b); //ok return 0; }
片段1中得出的结论:
当形参类型为基本数据类型时,非const形参可以接收const实参,同时也可以接收非const实参;
当形参类型为基本数据类型时,const形参可以接收const实参,同时也可以接收非const实参。
当形参为const参数时,不能在函数体内修改该形参的值。
代码片段2:
void add_1(int *a) { *a = *a +1; } void add_2(const int *a) { //*a = *a +2;//error,因为该指针指向const常量,不能修改常量的值 } int add_3(const int *a) { return *a +2; } int main() { int a = 1,b=2,c=3; const int a2 = 10; const int b2 = 20; const int c2 = 30; add_1(&a);//ok // add_1(&a2);//error add_3(&a);//ok add_3(&a2);//ok return 0; }
片段2中得出的结论:
当形参为const指针时,不能修改指针所指向的值。
当形参为非const指针时,只能接收非const指针实参,不能接收const指针实参。
当形参为const指针时,既可以接收非const指针实参, 也可以接收const指针实参。
代码片段3:
//引用形参,是实参的别名,修改的是实参的值 void add(int &x) { x = x +1 ; } //引用形参相当于额外的返回值 void fuc2(int i,int j,int &jf,int &jf2,int &cf,int &cf2) { jf = i + j; jf2 = i - j; cf = i * j; cf2 = i /j; } //在函数体内不需要修改的形参都应该设置为const &,即const引用 //指向地址的 引用 void cund(int *&x,int *&y) { int *temp = y; y = x; x = temp; } int main() { int i = 9; cout<<"before "<<i<<endl;//9 add(i); cout<<"after "<<i<<endl;//10 int j = 20; int v1,v2,v3,v4; fuc2(i,j,v1,v2,v3,v4); cout<<"加法结果:"<<v1<<endl; cout<<"减法结果:"<<v2<<endl; cout<<"乘法结果:"<<v3<<endl; cout<<"除法结果:"<<v4<<endl; cout<<"==========================="<<endl; int *ptri = &i;// ptri指针指向i int *ptrj = &j;// ptrj指针指向j cout<<"交换之前:"<<i <<" , "<<j<<endl; //i = 10 j = 20 cout<<"交换之前-指针:"<<*ptri <<" , "<<*ptrj<<endl; //*ptri = 10 *ptrj = 20 cund(ptri,ptrj);//交换的是指针的值,不是i,j的值 cout<<"交换之后:"<<i <<" , "<<j<<endl; //i = 10 j = 20 cout<<"交换之后-指针:"<<*ptri <<" , "<<*ptrj<<endl; //*ptri = 20 *ptrj = 20 return 0; }