函数参数传递方式

来源:互联网 发布:java反转单链表递归 编辑:程序博客网 时间:2024/05/22 06:39

函数参数传递方式之一:值传递

1)值传递的一个错误认识
先看考题一中Exchg1函数的定义:
void Exchg1(int x, int y) /*
定义中的x,y变量被称为Exchg1函数的形式参数 */
{
   inttmp;
   tmp = x;
   x = y;
   y = tmp;
   printf("x = %d, y = %d.\n", x, y);
}
问:你认为这个函数是在做什么呀?
答:好像是对参数xy的值对调吧?
请往下看,我想利用这个函数来完成对a,b两个变量值的对调,程序如下:
main()
{
   int a = 4,b = 6;
   Exchg1(a, b); /*a,b
变量为Exchg1函数的实际参数。*/
   printf("a = %d, b = %d.\n”, a, b);
   return(0);
}
我问:Exchg1()里头的printf("x= %d, y = %d.\n", x, y);语句会输出什么啊?我再问:Exchg1()后的printf("a = %d, b = %d.\n”, a, b);语句输出的是什么?
程序输出的结果是:
x = 6, y = 4.
a = 4, b = 6.
为什么不是a= 6,b = 4呢?奇怪,明明我把ab分别代入了xy中,并在函数里完成了两个变量值的交换,为什么ab变量值还是没有交换(仍然是a = 4b = 6,而不是a = 6b = 4)?如果你也会有这个疑问,那是因为你根本就不知实参ab与形参xy的关系了。

2)一个预备的常识
为了说明这个问题,我先给出一个代码:
   int a = 4;
   int x;
   x = a;
   x = x + 3;
看好了没,现在我问你:最终a值是多少,x值是多少?
(怎么搞的,给我这个小儿科的问题。还不简单,不就是a = 4x = 7嘛!)
在这个代码中,你要明白一个东西:虽然a值赋给了x,但是a变量并不是x变量哦。我们对x任何的修改,都不会改变a变量。呵呵!虽然简单,并且一看就理所当然,不过可是一个很重要的认识喔。

3)理解值传递的形式
看调用Exch1函数的代码:
main()
{
   int a = 4,b = 6;
   Exchg1(a, b) /*
这里调用了Exchg1函数 */
   printf("a = %d, b = %d.\n", a, b);
}
Exchg1(a, b)
时所完成的操作代码如下所示。
int x = a; /*
*/
int y = b; /*
注意这里,头两行是调用函数时的隐含操作 */
inttmp;
tmp = x;
x = y;
y = tmp;
请注意在调用执行Exchg1函数的操作中我人为地加上了头两句:
   int x = a;
   int y = b;
这是调用函数时的两个隐含动作。它确实存在,现在我只不过把它显式地写了出来而已。问题一下就清晰起来啦。(看到这里,现在你认为函数里面交换操作的是ab变量或者只是xy变量呢?)

原来,其实函数在调用时是隐含地把实参ab的值分别赋值给了xy,之后在你写的Exchg1函数体内再也没有对ab进行任何的操作了。交换的只是xy变量。并不是ab。当然ab的值没有改变啦!函数只是把ab的值通过赋值传递给了xy,函数里头操作的只是xy的值并不是ab的值。这就是所谓的参数的值传递了。

哈哈,终于明白了,正是因为它隐含了那两个的赋值操作,才让我们产生了前述的迷惑(以为ab已经代替了xy,对xy的操作就是对ab的操作了,这是一个错误的观点啊!)。

三、函数参数传递方式之二:地址传递

继续!地址传递的问题!
看考题二的代码:
void Exchg2(int *px, int *py)
{
   inttmp = *px;
   *px = *py;
   *py = tmp;
   printf("*px = %d, *py = %d.\n", *px, *py);
}
main()
{
   int a = 4;
   int b = 6;
   Exchg2(&a, &b);
   printf("a = %d, b = %d.\n”, a, b);
   return(0);
}
它的输出结果是:
*px = 6, *py = 4.
a = 6, b = 4.
看函数的接口部分:Exchg2(int*px, int *py),请注意:参数pxpy都是指针。再看调用处:Exchg2(&a, &b);
它将a的地址(&a)代入到pxb的地址(&b)代入到py。同上面的值传递一样,函数调用时作了两个隐含的操作:将&a&b的值赋值给了pxpy
   px = &a;
   py = &b;
呵呵!我们发现,其实它与值传递并没有什么不同,只不过这里是将ab的地址值传递给了pxpy,而不是传递的ab的内容,而(请好好地在比较比较啦)整个Exchg2函数调用是如下执行的:
   px = &a; /*
*/
   py = &b; /*
请注意这两行,它是调用Exchg2的隐含动作。*/
   inttmp = *px;
   *px = *py;
   *py = tmp;
   printf("*px =%d, *py = %d.\n", *px, *py);
这样,有了头两行的隐含赋值操作。我们现在已经可以看出,指针pxpy的值已经分别是ab变量的地址值了。接下来,对*px*py的操作当然也就是对ab变量本身的操作了。所以函数里头的交换就是对ab值的交换了,这就是所谓的地址传递(传递ab的地址给了pxpy),你现在明白了吗?

四、函数参数传递方式之三:引用传递

看题三的代码:
void Exchg3(int&x, int&y) /*
注意定义处的形式参数的格式与值传递不同 */
{
   inttmp = x;x = y;
   y = tmp;
   printf("x = %d, y = %d.\n", x, y);
}
main()
{
   int a = 4;
   int b = 6;
   Exchg3(a, b); /*
注意:这里调用方式与值传递一样*/
   printf("a = %d, b = %d.\n”, a, b);
}
输出结果:
x = 6, y = 4.
a = 6, b = 4. /*
这个输出结果与值传递不同。*/
看到没有,与值传递相比,代码格式上只有一处是不同的,即在定义处:
   Exchg3(int&x, int&y)
但是我们发现ab的值发生了对调。这说明了Exchg3(a, b)里头修改的是ab变量,而不只是修改xy了。

我们先看Exchg3函数的定义处Exchg3(int&x, int&y)。参数xyint的变量,调用时我们可以像值传递(如: Exchg1(a, b); )一样调用函数(如: Exchg3(a, b);)。但是xy前都有一个取地址符号“&”。有了这个,调用Exchg3时函数会将ab分别代替了xy了,我们称:xy分别引用了ab变量。这样函数里头操作的其实就是实参ab本身了,也就是说函数里是可以直接修改到ab的值了。

0 0
原创粉丝点击