回 帖 发 新 帖 刷新版面

主题:C指针不明者看《彻底搞定C指针》—第五篇:函数参数的传递

[center][b]彻底搞定C指针-——第五篇:函数参数的传递[/b][/center]

[b]一. 三道考题[/b]
开讲之前,我先请你做三道题目。(嘿嘿,得先把你的头脑搞昏才行……唉呀,谁扔我鸡蛋?)

[b]1. 考题一:[/b]程序代码如下:
void Exchg1(int x, int y)  
{
  int tmp;
  tmp=x;
  x=y;
  y=tmp;
  printf(“x=%d,y=%d\n”,x,y)
}
void main()
{
  int a=4,b=6;
  Exchg1 (a,b) ;
  printf(“a=%d,b=%d\n”,a,b)
}
[b]输出的结果[/b]:
x=____, y=____
a=____, b=____
问下划线的部分应是什么,请完成。

[b]2. 考题二:[/b]代码如下。
Exchg2(int *px, int *py)
{
  int tmp=*px;
  *px=*py;
   *py=tmp;
  print(“*px=%d,*py=%d\n”,*px,*py);
}
main()
{
  int a=4;
  int b=6;
      Exchg2(&a,&b);
      Print(“a=%d,b=%d\n”, a, b);
}
[b]输出的结果为[/b]:
*px=____, *py=____
a=____, b=____
问下划线的部分应是什么,请完成。

[b]3. 考题三:[/b]
Exchg2(int &x, int &y)
{
   int tmp=x;
   x=y;
   y=tmp;
  print(“x=%d,y=%d\n”,x,y);
}
main()
{
  int a=4;
  int b=6;
      Exchg2(a,b);
      Print(“a=%d,b=%d\n”, a, b);
}
[b]输出的结果:[/b]
x=____, y=____
a=____, b=____
    问下划线的部分输出的应是什么,请完成。

    你不在机子上试,能作出来吗?你对你写出的答案有多大的把握?
    正确的答案,想知道吗?(呵呵,让我慢慢地告诉你吧!)
    好,废话少说,继续我们的探索之旅了。
   我们都知道:C语言中函数参数的传递有:值传递,地址传递,引用传递这三种形式。题一为值传递,题二为地址传递,题三为引用传递。不过,正是这几种参数传递的形式,曾把我给搞得晕头转向。我相信也有很多人与我有同感吧?
下面请让我逐个地谈谈这三种传递形式。

[b]二. 函数参数传递方式之一:值传递[/b]

[b]1. 值传递的一个错误认识[/b]
    先看题一中Exchg1函数的定义:
void Exchg1(int x, int y)   //定义中的x,y变量被称为Exchg1函数的形式参数
{
  int tmp;
  tmp=x;
  x=y;
  y=tmp;
  printf(“x=%d,y=%d\n”,x,y)
}
[b]问:[/b]你认为这个函数是在做什么呀?
[b]答:[/b]好像是对参数x,y的值对调吧?
    请往下看,我想利用这个函数来完成对a,b两个变量值的对调,程序如下:
void main()
{
  int a=4,b=6;
  Exchg1 (a,b)     //a,b变量为Exchg1函数的实际参数。
/  printf(“a=%d,b=%d\n”,a,b)
}
    我问: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呢?

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

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

[b]3. 理解值传递的形式[/b]
看调用Exch1函数的代码:

main()
{
  int a=4,b=6;
  Exchg1(a,b) //这里调用了Exchg1函数        
  printf(“a=%d,b=%d”,a,b)
}

Exchg1(a,b)时所完成的操作代码如下所示。
int x=a;//←
int y=b;//←注意这里,头两行是调用函数时的隐含操作
int tmp;
tmp=x;
x=y;
y=tmp;
    请注意在调用执行Exchg1函数的操作中我人为地加上了头两句:
int x=a;
int y=b;
    这是调用函数时的两个隐含动作。它确实存在,现在我只不过把它显式地写了出来而已。问题一下就清晰起来啦。(看到这里,现在你认为函数里面交换操作的是a,b变量或者只是x,y变量呢?)
    原来 ,其实函数在调用时是隐含地把实参a,b 的值分别赋值给了x,y,之后在你写的Exchg1函数体内再也没有对a,b进行任何的操作了。交换的只是x,y变量。并不是a,b。当然a,b的值没有改变啦!函数只是把a,b的值通过赋值传递给了x,y,函数里头操作的只是x,y的值并不是a,b的值。这就是所谓的参数的值传递了。
    哈哈,终于明白了,正是因为它隐含了那两个的赋值操作,才让我们产生了前述的迷惑(以为a,b已经代替了x,y,对x,y的操作就是对a,b的操作了,这是一个错误的观点啊!)。

回复列表 (共46个回复)

沙发

[b]三. 函数参数传递方式之二:地址传递[/b]
继续——地址传递的问题!
看题二的代码:
Exchg2(int *px, int *py)
{
  int tmp=*px;
  *px=*py;
  *py=tmp;
  print(“*px=%d,*py=%d\n”,*px,*py);
}
main()
{
  int a=4;
  int b=6;
      Exchg2(&a,&b);
      Print(“a=%d,b=%d\n”, a, b);
}
[b]它的输出结果是[/b]:
  *px=6,*py=4
  a=6,b=4

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

[b]四. 函数参数传递方式之三:引用传递[/b]
    看题三的代码:
Exchg3(int &x, int &y) //注意定义处的形式参数的格式与值传递不同
{
   int tmp=x;
   x=y;
   y=tmp;
  print(“x=%d,y=%d\n”,x,y);
}
main()
{
  int a=4;
  int b=6;
      Exchg3(a,b);  //注意:这里调用方式与值传递一样
      Print(“a=%d,b=%d\n”, a, b);
}
[b]输出结果:[/b]
x=6, y=4
a=6, b=4   //这个输出结果与值传递不同。
    看到没有,与值传递相比,代码格式上只有一处是不同的,即在定义处:
Exchg3(int &x, int &y)。
    但是我们发现a与b的值发生了对调。这说明了Exchg3(a,b)里头修改的是a,b变量,而不只是修改x,y了。
    我们先看Exchg3函数的定义处Exchg3(int &x,int &y)。参数x,y是int的变量,调用时我们可以像值传递(如: Exchg1(a,b); )一样调用函数(如: Exchg3(a,b); )。但是x,y前都有一个取地址符号&。有了这个,调用Exchg3时函数会将a,b 分别代替了x,y了,我们称x,y分别引用了a,b变量。这样函数里头操作的其实就是实参a,b本身了,也就是说函数里是可以直接修改到a,b的值了。

   [b]最后对值传递与引用传递作一个比较:[/b]

[b]1. 在函数定义格式上有不同:[/b]
值传递在定义处是:Exchg1(int x, int y);
引用传递在这义处是:Exchg1(int &x, int &y);

[b]2. 调用时有相同的格式:[/b]
值传递:Exchg1(a,b);
引用传递:Exchg3(a,b);

[b]3. 功能上是不同的:[/b]
值传递的函数里操作的不是a,b变量本身,只是将a,b值赋给了x,y函数里操作的只是x,y变量而不是a,b,显示a,b的值不会被Exchg1函数所修改。
引用传递Exchg3(a,b)函数里是用a,b分别代替了x,y。函数里操作的就是a,b变量的本身,因此a,b的值可在函数里被修改的。

板凳

还行吧

3 楼

还不是很懂

4 楼

谢谢啦 明白的多啦 .今天晚上可以好好的睡觉了

5 楼

很好,说的很详细

6 楼

引用传递
    看题三的代码:
Exchg3(int &x, int &y) //注意定义处的形式参数的格式与值传递不同
{
   int tmp=x;
   x=y;
   y=tmp;
  print(“x=%d,y=%d\n”,x,y);
}
main()
{
  int a=4;
  int b=6;
      Exchg3(a,b);  //注意:这里调用方式与值传递一样
      Print(“a=%d,b=%d\n”, a, b);
}
输出结果:
x=6, y=4
a=6, b=4   //这个输出结果与值传递不同。
    看到没有,与值传递相比,代码格式上只有一处是不同的,即在定义处:
Exchg3(int &x, int &y)。
    但是我们发现a与b的值发生了对调。这说明了Exchg3(a,b)里头修改的是a,b变量,而不只是修改x,y了。
    我们先看Exchg3函数的定义处Exchg3(int &x,int &y)。参数x,y是int的变量,调用时我们可以像值传递(如: Exchg1(a,b); )一样调用函数(如: Exchg3(a,b); )。但是x,y前都有一个取地址符号&。有了这个,调用Exchg3时函数会将a,b 分别代替了x,y了,我们称x,y分别引用了a,b变量。这样函数里头操作的其实就是实参a,b本身了,也就是说函数里是可以直接修改到a,b的值了。

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

1. 在函数定义格式上有不同:
值传递在定义处是:Exchg1(int x, int y);
引用传递在这义处是:Exchg1(int &x, int &y);

2. 调用时有相同的格式:
值传递:Exchg1(a,b);
引用传递:Exchg3(a,b);

3. 功能上是不同的:
值传递的函数里操作的不是a,b变量本身,只是将a,b值赋给了x,y函数里操作的只是x,y变量而不是a,b,显示a,b的值不会被Exchg1函数所修改。
引用传递Exchg3(a,b)函数里是用a,b分别代替了x,y。函数里操作的就是a,b变量的本身,因此a,b的值可在函数里被修改的。




想与各位探讨C C++ VC Delphi Win32汇编 VB ASP等

 

非常感谢,以前没有看到过,可能是没有在意过这样的类型。今天记得了。

7 楼

好东西 我顶[em2][em2][em2]

8 楼

这个我勉强能懂,可是C语言这样让人生涩难懂的东西让人觉得象吃了个苍蝇,每当有点复杂的东西就要让人想半天到底是传啥东西,而且无法深入推敲,比如就照你说的例子:

我来添几个

4.
Exchg2(int *px, int *py)
Exchg2(a,b);

5.
Exchg2(int &x, int &y)
Exchg2(*a,*b);

这样的组合可以列出9种,虽然有些是错误通过不了编译的,但是其它的情况C大虾们有没有都穷举出来试一下把结果公布与众呢?如果别人写了一段程序在不知对错的情况下要你费尽脑子去分析,是不是很头晕啊!而且这样的标志&,*,->本身就不够严谨,如果数组a本身就是指的存放的首地址,那么&a是个什么东西!

所以我放弃了C!
Java没有指针功能一点不逊色于C!

9 楼

写的挺好!我也说一下我对这个题目的认识,大家多多指正!
形如[i]T& variable=other_variable;[/i]叫做变量绑定。也就是说,给一个变量在程序里面起一个别名。这个别名变量(variable)并不实际分配空间。而且一旦把他和其他变量在声明的时候被绑定,就无法拆开。也就是如果你的程序里面在声明的时候写上了:
int a;
int& i=a;
那么你在这两行后面就不能再把i变量重新赋值!
所以变量绑定必须在一开始就进行!

10 楼

呵呵!也不能说java一定就比cpp好。我认为,各有各的好处。适合c开发的东西不一定java就能行。比如说芯片级的东西!c语言的速度比较java要好的多(用汇编会增加编程难度,而且现在已经有一些芯片可支持SystemC)!因为java的运行需要虚拟机,而c不需要。不能因为java去掉了指针,就说明java比java就好得多。各有各的应用领域。我认为,我们应该培养思路,多学一些语言。在适合用c的地方用c,适合用就java的地方用java。语言就是我们思想的工具,是围绕我们的思路在转,不是我们的思路围绕语言在转。

我来回复

您尚未登录,请登录后再回复。点此登录或注册