指针传参
向函数传指针参数时,注意区分指针的指向(地址)
与 指针指向的地址中存放的值
的区别:
- 改变形参指针的指向,实参指针指向不会变,实参指向的地址存放的值也不会变;
- 改变形参指针指向的地址中存放的值,实参指针指向的地址中的值也会改变。
例1:改变形参指针的指向
例如,有个指针ptr1
指向变量a
。把ptr1
传给函数的一个形参指针ptr2
(这也是值传递,它所传递的是一个地址值)。如果在函数中将形参指针ptr2
指向了另外一个变量b
的地址,那也只是形参指针ptr2
的指向变了,不影响实参ptr1
。
void test(int *ptr2) {
int c = 2;
ptr2 = (&c);
}
int main() {
int a = 1;
int *ptr1 = &a;
test(ptr1);
std::cout << "*ptr1= " << *ptr1 << std::endl; //结果是 1
return 0;
}
可以看到,函数参数传了指针,形参指针改变指向,却并没有改变实参指针指向地址中的值。
例2:改变形参指针指向的地址中存放的值
下面这个传参传指针,但是函数中没有改变形参的指向,只改变了地址中的值,所以会改变实参指针指向地址中的值:
void test(int*ptr2){
(*ptr2)=3;
}
int main(){
int a=1,b=2;
int *ptr1=&a;
test(ptr1);
std::cout<< "*ptr1= " <<*ptr1<<std::endl; //结果是 3
return 0;
}
可以看到,函数参数传了指针,形参指针指向的地址中存放的值被修改了,(形参指针没有改变指向),实参指针指向地址中的值也被修改了。因为形参指针
和实参指针
都是指向
内存中同一块区域。
更详细的解释可参考:c++ 引用传参和指针传参的区别
指针参数传递:
- 指针参数传递本质上是值传递,它所传递的是一个地址值。值传递过程中,被调函数的形式参数作为被调函数的局部变量处理,会在栈中开辟内存空间以存放由主调函数传递进来的实参值,从而形成了实参的一个副本(替身)。值传递的特点是,被调函数对形式参数的任何操作都是作为局部变量进行的,不会影响主调函数的实参变量的值(形参指针变了,实参指针不会变)。
引用参数传递
-
引用参数传递过程中,被调函数的形式参数也作为局部变量在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参(本体)的任何操作都被处理成间接寻址,即通过栈中存放的地址访问主调函数中的实参变量(根据别名找到主调函数中的本体)。因此,被调函数对形参的任何操作都会影响主调函数中的实参变量。
-
引用传递和指针传递是不同的,虽然他们都是在被调函数栈空间上的一个局部变量,但是任何对于引用参数的处理都会通过一个
间接寻址
的方式操作到主调函数中的相关变量。而对于指针传递的参数,如果改变被调函数中的指针地址,它将应用不到主调函数的相关变量。如果想通过指针参数传递来改变主调函数中的相关变量(地址),那就得使用指向指针的指针或者指针引用。 -
从编译的角度来讲,程序在编译时分别将
指针
和引用
添加到符号表上,符号表中记录的是变量名及变量所对应地址。指针变量在符号表上对应的地址值为指针变量的地址值,而引用在符号表上对应的地址值为引用对象的地址值(与实参名字不同,地址相同)。符号表生成之后就不会再改,因此指针可以改变其指向的对象(指针变量中的值可以改),而引用对象则不能修改。