c语⾔函数传输传递的三种⽅式(值、指针、引⽤)本⽂摘⾃《彻底搞定c指针》
⼀、三道考题
开讲之前,我先请你做三道题⽬。(嘿嘿,得先把你的头脑搞昏才⾏……唉呀,谁扔我鸡蛋?)
考题⼀,程序代码如下:
void Exchg1(int x, int y)
{
int tmp;
tmp = x;
x = y;
y = tmp;
printf("x = %d, y = %d\n", x, y);
}
main()
{
int a = 4,b = 6;
Exchg1(a, b);
printf("a = %d, b = %d\n", a, b);
return(0);
}
输出的结果为: 20
x = ____, y=____.
a = ____, b=____.
问下划线的部分应是什么,请完成。
考题⼆,程序代码如下:
void Exchg2(int *px, int *py)
{
int tmp = *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=____, *py=____.
a=____, b=____.
问下划线的部分应是什么,请完成。
考题三,程序代码如下:
void Exchg3(int &x, int &y)
21
{
int tmp = 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);
return(0);
}
输出的结果为:
x=____, y=____.
a=____, b=____.
问下划线的部分应是什么,请完成。
你不在机⼦上试,能作出来吗?你对你写出的答案有多⼤的把握?
正确的答案,想知道吗?(呵呵,让我慢慢地告诉你吧!)
好,废话少说,继续我们的探索之旅了。
我们都知道:C语⾔中函数参数的传递有:值传递、地址传递、引⽤传递这三种形式。题⼀为值传递,题⼆为地址传递,题三为引⽤传递。不过,正是这⼏种参数传递的形式,曾把我给搞得晕头转向。我相信也有很多⼈与我有同感吧?
下⾯请让我逐个地谈谈这三种传递形式。
⼆、函数参数传递
1.⽅式⼀:值传递
(1)值传递的⼀个错误认识
先看考题⼀中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);
}
问:你认为这个函数是在做什么呀?
答:好像是对参数x、y的值对调吧?
请往下看,我想利⽤这个函数来完成对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);语
23
句会输出什么啊?
我再问:Exchg1()后的printf("a = %d, b = %d.\n”, a, b);语句输出的是什么?
程序输出的结果是:
x = 6, y = 4.
函数printfa = 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的关系了。
(2)⼀个预备的常识
为了说明这个问题,我先给出⼀个代码:
int a = 4;
int x;
x = a;
x = x + 3;
看好了没,现在我问你:最终a值是多少,x值是多少?
(怎么搞的,给我这个⼩⼉科的问题。还不简单,不就是a = 4、x = 7嘛!)
在这个代码中,你要明⽩⼀个东西:虽然a值赋给了x,但是a变量并不是x变量哦。我们对x任何的修改,都不会改变a变量。呵呵!虽然简单,并且⼀看就理所当然,不过可是⼀个很重要的认识喔。
(3)理解值传递的形式
24
看调⽤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; /* ←注意这⾥,头两⾏是调⽤函数时的隐含操作 */
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的值。这就是所谓的参数的值传递了。
哈哈,终于明⽩了,正是因为它隐含了那两个的赋值操作,才让我们产⽣
25
了前述的迷惑(以为a、b已经代替了x、y,对x、y的操作就是对a、b的操作了,这是⼀个错误的观点啊!
2. ⽅式⼆:地址传递
继续!地址传递的问题!
看考题⼆的代码:
void Exchg2(int *px, int *py)
{
int tmp = *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),请注意:参数px、py
3. ⽅式三:引⽤传递
看题三的代码:
void Exchg3(int &x, int &y) /* 注意定义处的形式参数的格式与值传递不同 */
{
int tmp = x;
27
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)
但是我们发现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的值了。
最后对值传递与引⽤传递作⼀个⽐较:
28
1)在函数定义格式上有不同:
值传递在定义处是:Exchg1(int x, int y);
引⽤传递在这义处是:Exchg3(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的值可在函数⾥被修改的。