C语言参数传递(值传递、地址传递)+二级指针
参数传递
C语言参数传递一般分为:值传递和地址传递(本质上只有值传递)
(注意:C语言中没有引用传递,C++才有引用传递,因为很多C语言环境是用C++编译器编译,使得C看起来支持引用传递,导致很多网上很多blog都把引用传递归为C语言参数传递的一类,
当使用Microsoft Visual C++ 2010 Express或VC6之类的编译器时使用引用传递就无法编译通过)
值传递:
形参是实参的拷贝,改变形参的值并不会影响外部实参的值。从被调用函数的角度来说,值传递是单向的(实参->形参)
1 #include 2 3 void swap(int x, int y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 9 swap(a, b);10 printf(“a=%d\nb=%d\n”, a, b);11 12 return 0;13 }14 void swap(int x, int y)15 {16 int t;17 18 t = x;19 x = y;20 y = t;21 }
上述代码运行后a、b的值并未改变
地址传递:
形参为指向实参地址的指针,当对形参的指向操作时,就相当于对实参本身进行的操作
1 #include 2 3 void swap(int *x, int *y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 9 swap(&a, &b);10 printf(“a=%d\nb=%d\n”, a, b);11 12 return 0;13 }14 void swap(int *x, int *y)15 {16 int t;17 18 t = *x;19 *x = *y;20 *y = t;21 }
上述代码执行后a和b值交换,a=20、b=10
易错点补充:
1 #include 2 3 void swap(int *x, int *y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 int *pp = &a; 9 int *kk = &b;10 swap(pp, kk);11 printf(“a=%d\nb=%d\n”, *pp, *kk);12 13 return 0;14 }15 void swap(int *x, int *y)16 {17 int *t;18 19 t = x;20 x = y;21 y = t;22 }
请读者想一下,上述代码执行后a和b的值是否交换,为什么?
上述代码看起来像交换了指针pp和kk的指向,实际上并没有
代码的运行结果a=10、b=20,运行结果是a和b的值并没有改变,因为这时使用的实参pp,kk是值传递,传递的是指针的值,以指针pp来说,指针的值是变量a的地址,
指针的值传入后用形参int *x和int *y接收,这里x和pp虽然都指向变量a的地址,但指针x和pp自身的地址并不相同(意思是x拷贝了一份pp的值),意味着你能改变变量a的值,但是不能改变pp的值(这里与值传递相似)
为了更加直观,清晰的看出值的交换,这里添加一些代码来显示内存和变量的值
1 #include 2 3 void swap(int *x, int *y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 int *pp = &a; 9 int *kk = &b;10 11 printf(“a的地址%p—-b的地址%p\n\n”, &a, &b);12 printf(“pp的值%p—-kk的值%p\n”, pp, kk);13 printf(“pp的地址%p—-kk的地址%p\n\n”, &pp, &kk);14 swap(pp, kk);15 printf(“a = %d\nb = %d”, *pp, *kk);16 17 return 0;18 }19 void swap(int *x, int *y)20 {21 int *t;22 23 printf(“x的值%p—-y的值%p\n”, x, y);24 printf(“x的地址%p—-y的地址%p\n”, &x, &y);25 26 t = x;27 x = y;28 y = t;29 }
从pp和x的地址可以看出,x和pp是两块不同的内存区域,x在swap函数内执行完后并不会对pp值产生任何影响,相当于复制了一份pp的值,如下图
(灵魂画手已上线)
传入指针的值虽然不能对指针的值进行修改,但是可以通过地址直接对a的值进行修改也可以实现交换,代码如下
1 #include 2 3 void swap(int *x, int *y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 int *pp = &a; 9 int *kk = &b;10 11 swap(pp, kk);12 printf(“a = %d\nb = %d”, *pp, *kk);13 14 return 0;15 }16 void swap(int *x, int *y)17 {18 int t;19 20 t = *x;21 *x = *y;22 *y = t;23 }
传入指针的值和和传入变量的地址在数值上是一样的,但是一个是传值传递一个地址传递,如下图
(灵魂画手已经上线)
二级指针
上面提到的参数传递方式有,传入变量的值,传入变量的地址,传入指针的值三种参数传递方式
下面介绍第四种,传入指针的地址,上面第二种(传入变量的地址)和第三种(传入指针的值)传递方式使用的用来接收参数的形参都是int *类型的指针
这里因为传入的是指针的地址,所以要使用二级指针int **x,因为传入的指针的地址,实现值的交换的方式就有两种一种是直接对变量的值进行修改,另一种就是对指针的值进行修改
第一种:直接对变量值进行修改
1 #include 2 3 void swap(int **x, int **y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 int *pp = &a; 9 int *kk = &b;10 11 swap(&pp, &kk);12 printf(“a = %d\nb = %d”, *pp, *kk);13 14 return 0;15 }16 void swap(int **x, int **y)17 {18 int t;19 20 t = **x;21 **x = **y;22 **y = t;23 }
第二种:对指针的值进行修改
1 #include 2 3 void swap(int **x, int **y); 4 5 main() 6 { 7 int a = 10, b = 20; 8 int *pp = &a; 9 int *kk = &b;10 11 swap(&pp, &kk);12 printf(“a = %d\nb = %d”, *pp, *kk);13 14 return 0;15 }16 void swap(int **x, int **y)17 {18 int *t;19 20 t = *x;21 *x = *y;22 *y = t;23 }
注意swap函数里用int *来交换指针的值,而不能用int **来进行交换,读者可自行修改代码验证想一下为什么?
下面是二级指针示意图
(灵魂画手已上线)
若有疑问,欢迎留言