赞
踩
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:C语言中函数参数传递的三种方式_c语言传参_魏波.的博客-CSDN博客
C语言中函数参数传递的三种方式
说几点建议:如果传值的话,会生成新的对象,花费时间和空间,而在退出函数的时候,又会销毁该对象,花费时间和空间。如果是你自己定义的类或结构等,都建议传指针或引用,因为他们不会创建新的对象。
例1:下面这段代码的输出结果为:
- #include<stdio.h>
- void change(int*a, int&b, int c)
- {
- c=*a;
- b=30;
- *a=20;
- }
- int main ( )
- {
- int a=10, b=20, c=30;
- change(&a,b,c);
- printf(“%d,%d,%d,”,a,b,c);
- return 0;
- }
- #include<stdio.h>
- void change(int*a, int&b, int c)
- {
- c=*a;
- b=30;
- *a=20;
- }
- int main ( )
- {
- int a=10, b=20, c=30;
- change(&a,b,c);
- printf(“%d,%d,%d,”,a,b,c);
- return 0;
- }
结果:20 30 30
解析:
该题考察函数传参问题。
1,指针传参 -> 将变量的地址直接传入函数,函数中可以对其值进行修改。
2,引用传参 -> 将变量的引用传入函数,效果和指针相同,同样函数中可以对其值进行修改。
3,值传参 -> 在传参过程中,首先将c的值复制给函数c变量,然后在函数中修改的即是函数的c变量,然后函数返回时,系统自动释放变量c。而对main函数的c没有影响。
例2:
- #include<stdio.h>
- void myswap(int x, int y)
- {
- int t;
- t=x;
- x=y;
- y=t;
- }
- int main()
- {
- int a, b;
- printf("请输入待交换的两个整数:");
- scanf("%d %d", &a, &b);
- myswap(a,b); //作为对比,直接交换两个整数,显然不行
- printf("调用交换函数后的结果是:%d 和 %d\n", a, b);
- return 0;
- }
-
-
- #include<stdio.h>
- void myswap(int *p1, int *p2)
- {
- int t;
- t=*p1;
- *p1=*p2;
- *p2=t;
- }
- int main()
- {
- int a, b;
- printf("请输入待交换的两个整数:");
- scanf("%d %d", &a, &b);
- myswap(&a,&b); //交换两个整数的地址
- printf("调用交换函数后的结果是:%d 和 %d\n", a, b);
- return 0;
- }
-
- #include<stdio.h>
- void myswap(int &x, int &y)
- {
- int t;
- t=x;
- x=y;
- y=t;
- }
-
- int main()
- {
- int a, b;
- printf("请输入待交换的两个整数:");
- scanf("%d %d", &a, &b);
- myswap(a,b); //直接以变量a和b作为实参交换
- printf("调用交换函数后的结果是:%d 和 %d\n", a, b);
- return 0;
- }
第一个的运行结果:输入2 3,输出2 3
第二个的运行结果:输入2 3,输出3 2
第三个的运行结果:输入2 3,输出3 2
解析:
在第一个程序中,传值不成功的原因是指在形参上改变了数值,没有在实参上改变数值。在第二个程序中,传地址成功的原因利用指针改变了原来的地址,所以实参就交换了。在第三个程序中,引用是直接改变两个实参变量a,b的值,所以就交换了。
下文会通过例子详细说明关于值传递,指针传递,引用传递
1)值传递:形参是实参的拷贝,改变形参的值并不会影响外部实参的值。从被调用函数的角度来说,值传递是单向的(实参->形参),参数的值只能传入,不能传出。当函数内部需要修改参数,并且不希望这个改变影响调用者时,采用值传递。
2)指针传递:形参为指向实参地址的指针,当对形参的指向操作时,就相当于对实参本身进行的操作
3)引用传递:形参相当于是实参的“别名”,对形参的操作其实就是对实参的操作,在引用传递过程中,被调函数的形式参数虽然也作为局部变量在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。
下面的代码对此作出了细致解释(从实参,形参在内存中存放地址的角度 说明了问题的本质,容易理解 )
- #include<iostream>
- using namespace std;
-
- //值传递
- void change1(int n){
- cout<<"值传递--函数操作地址"<<&n<<endl; //显示的是拷贝的地址而不是源地址
- n++;
- }
-
- //引用传递
- void change2(int & n){
- cout<<"引用传递--函数操作地址"<<&n<<endl;
- n++;
- }
-
- //指针传递
- void change3(int *n){
- cout<<"指针传递--函数操作地址 "<<n<<endl;
- *n=*n+1;
- }
-
- int main(){
- int n=10;
- cout<<"实参的地址"<<&n<<endl;
- change1(n);
- cout<<"after change1() n="<<n<<endl;
- change2(n);
- cout<<"after change2() n="<<n<<endl;
- change3(&n);
- cout<<"after change3() n="<<n<<endl;
-
- return true;
- }
运行结果如下,(不同的机器可能会有所差别)
可以看出,实参的地址为0x22ff44。采用值传递的时候,函数操作的地址是0x22ff20并不是实参本身,所以对它进行操作并不能改变实参的值。再看引用传递,操作地址就是实参地址 ,只是相当于实参的一个别名,对它的操作就是对实参的操作。接下来是指针传递,也可发现操作地址是实参地址。那么,引用传递和指针传递有什么区别吗?
引用的规则: 引用被创建的同时必须被初始化(指针则可以在任何时候被初始化)。 不能有NULL引用,引用必须与合法的存储单元关联(指针则可以是NULL)。 一旦引用被初始化,就不能改变引用的关系(指针则可以随时改变所指的对象)。
指针传递的实质:指针传递参数本质上是值传递的方式,它所传递的是一个地址值。值传递过程中,被调函数的形式参数作为被调函数的局部变量处理,即在栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。
指针传递和引用传递一般适用于: 函数内部修改参数并且希望改动影响调用者。对比指针/引用传递可以将改变由形参“传给”实参(实际上就是直接在实参的内存上修改,不像值传递将实参的值拷贝到另外的内存地址中才修改)。
另外一种用法是:当一个函数实际需要返回多个值,而只能显式返回一个值时,可以将另外需要返回的变量以指针/引用传递给函数,这样在函数内部修改并且返回后,调用者可以拿到被修改过后的变量,也相当于一个隐式的返回值传递吧。
从概念上讲。指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变。而引用是一个别名,它在逻辑上不是独立的,它的存在具有依附性,所以引用必须在一开始就被初始化,而且其引用的对象在其整个生命周期中是不能被改变的(自始至终只能依附于同一个变量)。
在C++中,指针和引用经常用于函数的参数传递,然而,指针传递参数和引用传递参数是有本质上的不同的:
为了进一步加深大家对指针和引用的区别,下面我从编译的角度来阐述它们之间的区别:
程序在编译时分别将指针和引用添加到符号表上,符号表上记录的是变量名及变量所对应地址。指针变量在符号表上对应的地址值为指针变量的地址值,而引用在符号表上对应的地址值为引用对象的地址值。符号表生成后就不会再改,因此指针可以改变其指向的对象(指针变量中的值可以改),而引用对象则不能修改。
最后,总结一下指针和引用的相同点和不同点:
1)相同点:
2)不同点:
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。