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);
}
输出的结果为:
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)
{
   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)值传递的一个错误认识
先看考题一中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);语句会输出什么啊?我再问: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的关系了。

(2)一个预备的常识
为了说明这个问题,我先给出一个代码:
 
 int a = 4;
   int x;
   x = a;
   x
= x + 3;
看好了没,现在我问你:最终a值是多少,x值是多少?
(怎么搞的,给我这个小儿科的问题。还不简单,不就是a = 4、x =
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; /* ← 注意这里,头两行是调用函数时的隐含操作
*/
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的操作了,这是一个错误的观点啊!)。

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


继续!地址传递的问题!
看考题二的代码:
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都是指针。再看调用处: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;
   printf("*px =%d, *py = %d.\n",
*px,
*py);
这样,有了头两行的隐含赋值操作。我们现在已经可以看出,指针px、py的值已经分别是a、b变量的地址值了。接下来,对*px、*py的操作当然也就是对a、b变量本身的操作了。所以函数里头的交换就是对a、b值的交换了,这就是所谓的地址传递(传递a、b的地址给了px、py),你现在明白了吗?

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


看题三的代码:
void Exchg3(int &x, int &y) /* 注意定义处的形式参数的格式与值传递不同
*/
{
   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);
}
输出结果:
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);
引用传递在这义处是: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的值可在函数里被修改的。

C语言的参数传递

时间: 2024-08-28 13:25:50

C语言的参数传递的相关文章

C语言之参数传递

学了四年的计算机,一直让自己比较苦恼的问题是C语言的参数传递问题,之所以说是苦恼,是因为在某年的一个学期,不幸接触到数据结构,光一个链表就把自己弄得死去活来的,而且自已一直就楞以为在操作的过程中,传递参数的指针也在发生变化,结果可想而知,数据结构成了自己专业课里面险些挂彩的一门课程. 直至最近,拿出数据结构打算恶补一下前些年欠下的债,第一件事情,还是先解决好当时困扰自己半年之久的C语言参数传递问题吧. C语言课堂上,自己明明记得参数传递问题我已经搞懂了,只不过当时没有接触过C plus plus

c语言 可变参数传递 va_list使用

通过使用VA_LIST可以实现向函数传递不同数目的参数. #include <stdarg.h> #include <iostream> #include <string> using namespace std; #pragma argsused //函数A 传递若干个整形变量 void Funca(int n,...) {  //定义获取变量的结构体  va_list va_ptr;  //开始从头部开始获取变量  va_start(va_ptr,n);  for(

C语言函数参数传递原理

C语言中参数的传递方式一般存在两种方式:一种是通过栈的形式传递,另一种是通过寄存器的方式传递的.这次,我们只是详细描述一下第一种参数传递方式,另外一种方式在这里不做详细介绍. 首先,我们看一下,下面一个简单的调用例程: int Add (int a, int b, int c) { return a+b+c; } void main() { int x =0 , y = 1, z = 2; int result = 0; result = Add(x, y, z); printf("Result

菜鸟学习-C语言函数参数传递详解-结构体与数组

C语言中结构体作为函数参数,有两种方式:传值和传址. 1.传值时结构体参数会被拷贝一份,在函数体内修改结构体参数成员的值实际上是修改调用参数的一个临时拷贝的成员的值,这不会影响到调用参数.在这种情况下,涉及到结构体参数的拷贝,程序空间及时间效率都会受到影响. 例子: typedef struct tagSTUDENT{ char name[20]; int age; }STUDENT; void fun(STUDENT stu) { printf("stu.name=%s,stu.age=%d/

参数传递

java语言的参数传递问题 基本数据类型的参数传递,形式参数的改变对实际参数没有影响,因为传递的是具体的数值 引用数据类型的参数传递,形式参数的改变对实际参数有影响,因为传递的是地址值 ; ( String 例外) java中到底是传值还是传地址 1.既是传值,也是传地址,基本数据类型传递的值,引用数据类型传递的地址 2.java中只有传值,因为地址也是值(出去面试都说这种,支持者是高斯林java之父)

数据结构C语言实现——线性链表

declaration.h #ifndef DECLARATION_H_INCLUDED #define DECLARATION_H_INCLUDED #define TRUE 1 #define FALSE 0 #define OK 1 #define ERROR 0 #define INFEASIBLE -1 #define OVERFLOW -2 #define ElemType int typedef ElemType* Triplet; typedef int Status; type

参数传递问题

一直没有好好研究C语言参数传递,只是从书上看,知道有传值和传地址两种方式,有时候一知半解真的不行,看似懂了,遇到问题后就乱了手脚,他们内部发生什么变化并不清楚,现在写下来.C语言的参数传递有传值和传地址两种方式.传值的过程:(1)行参与实参各占一个独立的存储空间.(2)行参的存储空间是函数被调用时才分配的.调用开始,系统为行参开辟一个临时存储区,然后将各实参之值传递给行参,这时行参就得到了实参的值.(3)函数返回时,临时存储区也被撤销.传值的特点:单向传递,即函数中对行参变量的操作不会影响到调用

方法重载和参数传递

-方法重载和参数传递 方法相同:在 Java 语言中,方法相同的概念和其它程序设计语言不尽相同,Java语言中的方法相同指方法名称和参数列表都相同,其中参数列表相同指参数个数.参数类型和参数排列顺序等相同,参数名称可以不相同.相同的方法访问控制符.返回值类型可以不相同.以下是一下相同的方法:public void test(int a,double[] d)private int test(int i,double[] d1)在同一个类内部,不能声明相同的方法,否则将出现语法错误. 方法重载:

Object Pascal 语法之语言基础(四)

1.8 过程与函数 过程与函数是实现一定功能的语句块,是程序中的特定功能单元.可以在程序的其他地方被调用,也可以进行递归调用.过程与函数的区别在于过程没有返回值,而函数有返回值. 1.过程与函数的定义过程与函数的定义包括过程原型或函数原型.过程体或函数体的定义.过程定义的形式如下: procedure ProcedureName(ParameterList); directives; var LocalDeclarations; begin statements end; ProcedureNa