主题:C指针不明者看《彻底搞定C指针》—第五篇:函数参数的传递
白云小飞
[专家分:260] 发布于 2005-09-09 11:52:00
[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个回复)
11 楼
yunying5241 [专家分:0] 发布于 2005-09-14 09:09:00
你的还是有问题
第2个地址传递的a=4,b=6并不应互换
12 楼
wangyunzhou [专家分:810] 发布于 2005-09-14 10:17:00
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);
}
输出的结果:
x=____, y=____
a=____, b=____
这样可以吗??参数型号会不会不符??
13 楼
wangyunzhou [专家分:810] 发布于 2005-09-14 10:58:00
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);
}
输出的结果:
x=____, y=____
a=____, b=____
这样也可以吗 不出错吗?形参是地址,而实参是数
14 楼
白云小飞 [专家分:260] 发布于 2005-09-14 16:13:00
****************
[b]楼8的问题:[/b]
4.
Exchg2(int *px, int *py)
Exchg2(a,b);
5.
Exchg2(int &x, int &y)
Exchg2(*a,*b);
你举的4,5两个组合,并有9种组合,认为太复杂了。其实不然:
4的情况:px 和py都是int * 的类型,而a,b是int 类型,实参与形参数据类型是不一样的,当然是不能代入的,并毫无意义,因此是不用考虑。
5的情况及其它的情况也都是一样的问题。
**********************
[b]楼11的问题:第2个地址传递的a=4,b=6并不应互换[/b]
a,b变量的值是会互换的。你可能对*px,*py操作的是什么不太明白。
*********************
[b]楼13:[/b]
很好,你提出的问题正是我们初接触“引用传递”时容易迷惑的问题:
Exchg3(int &x, int &y)
{
……
}
定义中的形参是x而不是&x,x是int形变量而不是地址。
实参a也是int变量,因此类型相同。
但是在形参前有一个&的符号,它的意思是我要直接引用实参本身,而不是值传递,即用a,b分别代替x,y这个意思。
即:
调用Exchg3(a,b)时,Exchg3所完成的操作如下:
int &x=a;
int &y=b; //将x绑定到a,将y绑定到b.
int tmp=x;
x=y;
y=tmp;
因为绑定了,对形参x,y的操作就是对实参a,b的操作。
(感谢9楼给我的概念!请看9楼对int &x=a的解说。)
(也谢谢各位的讨论了!)
15 楼
Flieder [专家分:160] 发布于 2005-09-14 21:27:00
呵呵,俺再来说一段我对引用变量(也就是T&)的见解:
首先,我们来看一段代码:
#include <cstdlib>
#include <iostream>
using namespace std;
int test(int& a,int& b)
{
return a*b;
}
int main(int argc, char *argv[])
{
printf("%d",test(1,2));
system("PAUSE");
return EXIT_SUCCESS;
}
大家认为这段代码是否能够运行呢?呵呵,答案是不能。
因为引用变量无法和一个数值邦定在一起,他必须和另外一个变量邦定在一起。尽管有些书上说可以把引用变量和一个数值邦定在一起,内部工作机制上是,首先在内存中开辟一个临时空间,然后把引用变量和这个临时空间邦定。但是经过我的测试,这是错误的。顺便说一下,我用的是DEV C++(内核是gcc),这段程序是c++语法。
这段程序改称如下的代码则可运行:
#include <cstdlib>
#include <iostream>
using namespace std;
int test(int& a,int& b)
{
return a*b;
}
int main(int argc, char *argv[])
{
int x=1,y=2;
printf("%d",test(x,y));
system("PAUSE");
return EXIT_SUCCESS;
}
还有一个有趣现象。cpp里面有函数重载的这个功能。
那么我们我们继续做下列测试,看看是否能顺利运行:
#include <cstdlib>
#include <iostream>
using namespace std;
int test(int& a,int& b)
{
return a*b;
}
int test(int a,int b)
{
return a*b;
}
int main(int argc, char *argv[])
{
int x=1,y=2;
printf("%d",test(x,y));
system("PAUSE");
return EXIT_SUCCESS;
}
呵呵,大家估计的结果如何呢?
其实这段代码也不能运行。这下大家奇怪了,为什么不能运行,int test(int& a,int& b)和int test(int a,int b)在参数上面是不同的,两个函数应该是重载关系。可是实际上在int test(int& a,int& b)这个函数里面两个参数实际上是两个整形
变量的别名,他们不分配存储空间,所以用的也相当是被绑定的变量作为形参。所以,这两个函数形异实同,不能同时出现。大家要注意这个问题!
16 楼
田园的拾荒者 [专家分:0] 发布于 2005-09-22 02:18:00
明白了`终于明白了 `不过还是要复习 啊`~
17 楼
CCCP [专家分:2290] 发布于 2005-09-22 12:16:00
因为引用变量无法和一个数值邦定在一起,他必须和另外一个变量邦定在一起。尽管有些书上说可以把引用变量和一个数值邦定在一起,内部工作机制上是,首先在内存中开辟一个临时空间,然后把引用变量和这个临时空间邦定。但是经过我的测试,这是错误的。
========================================
临时变量不能做左值,所以2没办法隐式转换成int&,但是const int&就可以
#include <cstdlib>
#include <iostream>
using namespace std;
int test(const int& a, const int& b)
{
return a*b;
}
int main(int argc, char *argv[])
{
printf("%d",test(1,2));
system("PAUSE");
return EXIT_SUCCESS;
}
当然对于基本类型来说,这样做是愚蠢的,但是实际应用中,这种做法是相当多的,比如
size_t test(const std::string& str) {
return str.length();
}
std::string s1("aaaaa");
const char* s2 = "aaaaa";
char s3[256];
sprintf(s3, "aaaaa");
test(s1);
test(s2);
test(s3);
test("aaaaa");
18 楼
lovelylan [专家分:0] 发布于 2005-09-24 11:30:00
“抄书”还是“原创”?
抄书鄙视!
原创支持!
不过我也看不懂。。。。
19 楼
忧愁虾米 [专家分:600] 发布于 2005-09-24 15:44:00
其实基础好的人完全明白...
哈哈
我基础好,我明白
自大狂,我鄙视我自己
20 楼
我是岳阳人 [专家分:1530] 发布于 2005-09-24 17:46:00
Exchg2(int &x, int &y) //这么定义有问题吗,我很少见到哦
我来回复