语言提升——函数参数的传递



函数的参数传递

定义了int a=10,b=20;Exchange函数实现传入的参数对换值的功能,调用之后看a,b的值为多少。

1、值传递

 函数定义:void Exchange1( int x , int y );

 函数调用:Exchange1( a , b );

2、地址传递

  函数定义:void Exchange2( int *px , int *py );

 函数调用:void Exchange2( &a , &b );

3、引用传递(c++的特性,在我的文件为.cpp能用)

  函数定义:void Exchange3( int &x , int &y );

  函数调用:Exchange3( a , b );

如果不太清晰上面的结果与分析,可看看下面的讲解。



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

(1)值传递的一个错误认识

#include <stdio.h>

void Exchange1(int x, int y) // 定义中的x,y变量被称为Exchange1函数的形式参数
{
    int tmp;
    tmp = x;
    x = y;
    y = tmp;
    printf("x = %d, y = %d\n", x, y);
}
//问:你认为这个函数是在做什么呀?
//答:好像是对参数x、y的值对调吧?
//请往下看,我想利用这个函数来完成对a,b两个变量值的对调,程序如下:
int main()
{
    int a = 10,b = 20;
    Exchange1(a, b); //a,b变量为Exchange1函数的实际参数.
    printf("a = %d, b = %d\n" ,a, b);
    return 0;
}
//我问:Exchange1()里头的printf("x = %d, y = %d.\n", x, y);语句会输出什么啊?
//我再问:Exchange1()后的printf("a = %d, b = %d.\n”, a, b);语句输出的是什么?

程序输出的结果是: 

x = 20, y = 10

a = 10, b =20

为什么不是a = 20,b = 10呢? 

奇怪,明明我把a、b分别代入了x、y中,并在函数里完成了两个变量值的交换,为什么a、b变量值还是没有交换(仍然是a = 10、b = 20,而不是a = 20、b = 10)?如果你也会有这个疑问,那是因为你根本就不知实参a、b与形参x、y的关系了。 

(2)一个预备的常识

为了说明这个问题,我先给出一个代码: 

int a = 4; 

int x; 

x = a; 

x = x + 3; 

看好了没,现在我问你:最终a值是多少,x值是多少? 

(怎么搞的,给我这个小儿科的问题。还不简单,不就是a = 4、x = 7嘛!) 

在这个代码中,你要明白一个东西:虽然a值赋给了x,但是a变量并不是x变量哦。我们对x任何的修改,都不会改变a变量。呵呵!虽然简单,并且一看就理所当然,不过可是一个很重要的认识喔。 

(3)理解值传递的形式 

看调用Exchange1函数的代码: 

main() 

int a = 10,b = 20; 

Exchange1(a, b) /* 这里调用了Exchange1函数 */ 

printf("a = %d, b = %d.\n", a, b); 

Exchange1(a, b)时所完成的操作代码如下所示。 

int x = a; /* ← */ 

int y = b; /* ← 注意这里,头两行是调用函数时的隐含操作 */ 

int tmp; 

tmp = x; 

x = y; 

y = tmp; 

请注意在调用执行Exchange1函数的操作中我人为地加上了头两句: 

int x = a; 

int y = b; 

这是调用函数时的两个隐含动作。它确实存在,现在我只不过把它显式地写了出来而已。问题一下就清晰起来啦。(看到这里,现在你认为函数里面交换操作的是a、b变量或者只是x、y变量呢?) 

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

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


2. 函数参数传递方式之二:地址传递

继续!地址传递的问题! 

#include <stdio.h>

void Exchange2(int *px, int *py)
{
    int tmp = *px;
    *px = *py;
    *py = tmp;
    printf("*px = %d, *py = %d\n", *px, *py);
}

int main()
{
    int a = 10;
    int b = 20;
    Exchange2(&a, &b);
    printf("a = %d, b = %d\n",a, b);
    return 0;
}

它的输出结果是: 

*px = 20, *py = 10

a = 20, b = 10

看函数的接口部分:Exchange2(int *px, int *py),请注意:参数px、 py都是指针。 

再看调用处:Exchange2(&a, &b); 

它将a的地址(&a)代入到px,b的地址(&b)代入到py。同上面的值传递一样,函数调用时作了两个隐含的操作:将&a,&b的值赋值给了px、py。 

px = &a; 

py = &b; 

呵呵!我们发现,其实它与值传递并没有什么不同,只不过这里是将a、b的地址值传递给了px、py,而不是传递的a、b的内容,而(请好好地在比较比较啦)整个Exchange2函数调用是如下执行的: 

px = &a; /* ← */ 

py = &b; /* ← 请注意这两行,它是调用Exchange2的隐含动作。*/ 

int tmp = *px; 

*px = *py; 

*py = tmp; 

printf("*px =%d, *py = %d.\n", *px, *py); 

这样,有了头两行的隐含赋值操作。我们现在已经可以看出,指针px、py的值已经分别是a、b变量的地址值了。接下来,对*px、*py的操作当然也就是对a、b变量本身的操作了。所以函数里头的交换就是对a、b值的交换了,这就是所谓的地址传递(传递a、b的地址给了px、py),你现在明白了吗? 


3. 函数参数传递方式之三:引用传递

看题三的代码: c++环境下编译运行

#include <stdio.h>

void Exchange3(int &x, int &y) // 注意定义处的形式参数的格式与值传递不同
{
    int tmp = x;
    x = y;
    y = tmp;
    printf("x = %d, y = %d\n", x, y);
}

int main()
{
    int a = 10;
    int b = 20;
    Exchange3(a, b); // 注意:这里调用方式与值传递一样
    printf("a = %d, b = %d\n", a, b);
    return 0;
}

输出结果: 

x = 20, y = 10

a = 20, b = 10  // 这个输出结果与值传递不同。

看到没有,与值传递相比,代码格式上只有一处是不同的,即在定义处: 

Exchange3( int &x ,  int  &y ) 

但是我们发现a与b的值发生了对调。这说明了Exchange3(a, b)里头修改的是a、b变量,而不只是修改x、y了。 

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

最后对值传递与引用传递作一个比较: 

1)在函数定义格式上有不同: 

值传递在定义处是:Exchange1(int x, int y); 

引用传递在这义处是:Exchange3(int &x, int &y); 

2)调用时有相同的格式: 

值传递:Exchange1(a, b); 

引用传递:Exchange3(a, b); 

3)功能上是不同的: 

值传递的函数里操作的不是a、b变量本身,只是将a、b值赋给了x、y。函数里操作的只是x、y变量而不是a、b,显示a、b的值不会被Exchange1函数所修改。 

引用传递Exchange3(a, b)函数里是用a、b分别代替了x、y。函数里操作的就是a、b变量的本身,因此a、b的值可在函数里被修改的。 



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值