值传递、指针传递、引用传递

</pre><p></p><p>原文:<a target=_blank href="http://www.cnblogs.com/yanlingyin/archive/2011/12/07/2278961.htmlhttp://">http://www.cnblogs.com/yanlingyin/archive/2011/12/07/2278961.html</a></p><p></p><p>最近写了几篇深层次讨论数组和指针的文章,其中提到了“C语言中,所有非数组的形式参数传递均以值传递形式”</p><h1 class="postTitle"><a target=_blank id="ctl02_TitleUrl" class="postTitle2" href="http://www.cnblogs.com/yanlingyin/archive/2011/12/06/2277821.html">数组和指针背后——内存角度</a></h1><h1 class="postTitle"><a target=_blank id="ctl02_TitleUrl" class="postTitle2" href="http://www.cnblogs.com/yanlingyin/archive/2011/11/29/2268391.html">语义"陷阱"---数组和指针</a></h1><p>而关于值传递,指针传递,引用传递这几个方面还会存在误区, 所有我觉的有必要在这里也说明一下~</p><p>下文会通过例子详细说明哦</p><p><strong>值传递:</strong></p><p><strong></strong>形参是实参的拷贝,改变形参的值并不会影响外部实参的值。从被调用函数的角度来说,值传递是单向的(实参->形参),参数的值只能传入,</p><p>不能传出。当函数内部需要修改参数,并且不希望这个改变影响调用者时,采用值传递。</p><p><strong>指针传递:</strong></p><p>形参为指向实参地址的指针,当对形参的指向操作时,就相当于对实参本身进行的操作</p><p><strong>引用传递:</strong></p><p>形参相当于是实参的“别名”,对形参的操作其实就是对实参的操作,在引用传递过程中,被调函数的形式参数虽然也作为局部变量在栈</p><p>中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过</p><p>栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。</p><p> </p><p>理论性的就不多说,</p><p>下面的代码对此作出了细致解释(从实参,形参在内存中存放地址的角度 说明了问题的本质,容易理解  )</p><div class="cnblogs_code"><div>按 Ctrl+C 复制代码</div><textarea style="width: 909px; height: 467.2px; font-family: Courier New; font-size: 12px; line-height: 1.5;"></textarea><div>按 Ctrl+C 复制代码</div></div><p>运行结果如下,(不同的机器可能会有所差别)</p><p><img src="http://pic002.cnblogs.com/images/2011/348708/2011120710381445.jpg" alt="" /></p><p>可以看出,实参的地址为0x22ff44</p><p>采用值传递的时候,函数操作的地址是0x22ff20并不是实参本身,所以对它进行操作并不能改变实参的值</p><p>再看引用传递,操作地址就是实参地址 ,只是相当于实参的一个别名,对它的操作就是对实参的操作</p><p>接下来是指针传递,也可发现操作地址是实参地址</p><p>那么,引用传递和指针传递有什么区别吗?</p><p><strong>  引用的规则: </strong>(1)引用被创建的同时必须被初始化(指针则可以在任何时候被初始化)。 </p><p>(2)不能有NULL引用,引用必须与合法的存储单元关联(指针则可以是NULL)。 (3)一旦引用被初始化,就不能改变引用的关系(指针则可以随时改变所指的对象)。 </p><p><strong>指针传递的实质:</strong></p><p>指针传递参数本质上是值传递的方式,它所传递的是一个地址值。值传递过程中,被调函数的形式参数作为被调函数的局部变量处理,</p><p>即在栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的</p><p>任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值<span>。<span style="color:#ff0000;">(这里是在说实参指针本身的地址值不会变)<span style="color:#000000;">如果理解不了大可跳过这段</span></span></span></p><p><strong>指针传递和引用传递一般适用于</strong>:</p><p>函数内部修改参数并且希望改动影响调用者。对比指针/引用传递可以将改变由形参“传给”实参(实际上就是直接在实参的内存上修改,</p><p>不像值传递将实参的值拷贝到另外的内存地址中才修改)。</p><p>另外一种用法是:当一个函数实际需要返回多个值,而只能显式返回一个值时,可以将另外需要返回的变量以指针/引用传递</p><p>给函数,这样在函数内部修改并且返回后,调用者可以拿到被修改过后的变量,也相当于一个隐式的返回值传递吧。</p><p> </p><p> </p><p><strong>以下是我觉得关于指针和引用写得很不错的文章,大家可参照看一下,原文出处地址</strong>:<a target=_blank href="http://xinklabi.iteye.com/blog/653643">http://xinklabi.iteye.com/blog/653643</a> </p><p>从概念上讲。指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变。</p><p>而引用是一个别名,它在逻辑上不是独立的,它的存在具有依附性,所以引用必须在一开始就被初始化,而且其引用的对象在其整个生命周期中是不能被改变的(自始至终只能依附于同一个变量)。</p><p>在<span>C++</span>中,指针和引用经常用于函数的参数传递,然而,指针传递参数和引用传递参数是有本质上的不同的:</p><p>指针传递参数本质上是值传递的方式,它所传递的是一个地址值。值传递过程中,被调函数的形式参数作为被调函数的局部变量处理,即在栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值<span>。(这里是在说实参指针本身的地址值不会变)</span></p><p>而在引用传递过程中,被调函数的形式参数虽然也作为局部变量在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。</p><p>引用传递和指针传递是不同的,虽然它们都是在被调函数栈空间上的一个局部变量,但是任何对于引用参数的处理都会通过一个间接寻址的方式操作到主调函数中的相关变量。而对于指针传递的参数,如果改变被调函数中的指针地址,它将影响不到主调函数的相关变量。如果想通过指针参数传递来改变主调函数中的相关变量,那就得使用指向指针的指针,或者指针引用。</p><p><span style="color:#ff0000;">为了进一步加深大家对指针和引用的区别,下面我从编译的角度来阐述它们之间的区别:</span></p><p><span style="color:#ff0000;">程序在编译时分别将指针和引用添加到符号表上,符号表上记录的是变量名及变量所对应地址。指针变量在符号表上对应的地址值为指针变量的地址值,而引用在符号表上对应的地址值为引用对象的地址值。符号表生成后就不会再改,因此指针可以改变其指向的对象(指针变量中的值可以改),而引用对象则不能修改。</span></p><p>最后,总结一下指针和引用的相同点和不同点:</p><p>★相同点:</p><p>●都是地址的概念;</p><p>指针指向一块内存,它的内容是所指内存的地址;而引用则是某块内存的别名。</p><p>★不同点:</p><p>●指针是一个实体,而引用仅是个别名;</p><p>●引用只能在定义时被初始化一次,之后不可变;指针可变;引用“从一而终”,指针可以“见异思迁”;</p><p>●引用没有<span>const</span>,指针有<span>const</span>,<span>const</span>的指针不可变;(<span>具体指没有int& const a这种形式,而const int& a是有     的,  前者指引用本身即别名不可以改变,这是当然的,所以不需要这种形式,后者指引用所指的值不可以改变</span>)</p><p>●引用不能为空,指针可以为空;</p><p>●“<span>sizeof </span>引用”得到的是所指向的变量<span>(</span>对象<span>)</span>的大小,而“<span>sizeof </span>指针”得到的是指针本身的大小;</p><p>●指针和引用的自增<span>(++)</span>运算意义不一样;</p><p>●引用是类型安全的,而指针不是<span> (</span>引用比指针多了类型检查)</p><div class="cnblogs_code"><div class="cnblogs_code_toolbar"><span class="cnblogs_code_copy"><a target=_blank title="复制代码"><img src="https://i-blog.csdnimg.cn/blog_migrate/69c5a8ac3fa60e0848d784a6dd461da6.gif" alt="复制代码" /></a></span></div><pre>一、引用的概念

引用引入了对象的一个同义词。定义引用的表示方法与定义指针相似,只是用&代替了*。
例如: Point pt1(<span style="color:#800080;">10</span>,<span style="color:#800080;">10</span>);
Point &pt2=pt1; 定义了pt2为pt1的引用。通过这样的定义,pt1和pt2表示同一对象。
需要特别强调的是引用并不产生对象的副本,仅仅是对象的同义词。因此,当下面的语句执行后:
pt1.offset(<span style="color:#800080;">2</span>,<span style="color:#800080;">2</span>);
pt1和pt2都具有(<span style="color:#800080;">12</span>,<span style="color:#800080;">12</span>)的值。
引用必须在定义时马上被初始化,因为它必须是某个东西的同义词。你不能先定义一个引用后才
初始化它。例如下面语句是非法的:
Point &pt3;
pt3=pt1;
那么既然引用只是某个东西的同义词,它有什么用途呢?
下面讨论引用的两个主要用途:作为函数参数以及从函数中返回左值。 

二、引用参数

<span style="color:#800080;">1</span>、传递可变参数
传统的c中,函数在调用时参数是通过值来传递的,这就是说函数的参数不具备返回值的能力。
所以在传统的c中,如果需要函数的参数具有返回值的能力,往往是通过指针来实现的。比如,实现
两整数变量值交换的c程序如下:
<span style="color:#0000ff;">void</span> swapint(<span style="color:#0000ff;">int</span> *a,<span style="color:#0000ff;">int</span> *b)
{
<span style="color:#0000ff;">int</span> temp;
temp=*a;
a=*b;
*b=temp;
}

使用引用机制后,以上程序的c++版本为:
<span style="color:#0000ff;">void</span> swapint(<span style="color:#0000ff;">int</span> &a,<span style="color:#0000ff;">int</span> &b)
{
<span style="color:#0000ff;">int</span> temp;
temp=a;
a=b;
b=temp;
}
调用该函数的c++方法为:swapint(x,y); c++自动把x,y的地址作为参数传递给swapint函数。

<span style="color:#800080;">2</span>、给函数传递大型对象
当大型对象被传递给函数时,使用引用参数可使参数传递效率得到提高,因为引用并不产生对象的
副本,也就是参数传递时,对象无须复制。下面的例子定义了一个有限整数集合的类: 
<span style="color:#0000ff;">const</span> maxCard=<span style="color:#800080;">100</span>; 
Class Set 
{
<span style="color:#0000ff;">int</span> elems[maxCard]; <span style="color:#008000;">//</span><span style="color:#008000;"> 集和中的元素,maxCard 表示集合中元素个数的最大值。 </span><span style="color:#008000;">
</span><span style="color:#0000ff;">int</span> card; <span style="color:#008000;">//</span><span style="color:#008000;"> 集合中元素的个数。 </span><span style="color:#008000;">
</span><span style="color:#0000ff;">public</span>:
Set () {card=<span style="color:#800080;">0</span>;} <span style="color:#008000;">//</span><span style="color:#008000;">构造函数</span><span style="color:#008000;">
</span>friend Set <span style="color:#0000ff;">operator</span> * (Set ,Set ) ; <span style="color:#008000;">//</span><span style="color:#008000;">重载运算符号*,用于计算集合的交集 用对象作为传值参数
</span><span style="color:#008000;">//</span><span style="color:#008000;"> friend Set operator * (Set & ,Set & ) 重载运算符号*,用于计算集合的交集 用对象的引用作为传值参数 </span><span style="color:#008000;">
</span>...
}
先考虑集合交集的实现
Set <span style="color:#0000ff;">operator</span> *( Set Set1,Set Set2)
{
Set res;
<span style="color:#0000ff;">for</span>(<span style="color:#0000ff;">int</span> i=<span style="color:#800080;">0</span>;i<Set1.card;++i)
<span style="color:#0000ff;">for</span>(<span style="color:#0000ff;">int</span> j=<span style="color:#800080;">0</span>;j>Set2.card;++j)
<span style="color:#0000ff;">if</span>(Set1.elems[i]==Set2.elems[j])
{
res.elems[res.card++]=Set1.elems[i];
<span style="color:#0000ff;">break</span>;
}
<span style="color:#0000ff;">return</span> res;
}
由于重载运算符不能对指针单独操作,我们必须把运算数声明为 Set 类型而不是 Set * 。
每次使用*做交集运算时,整个集合都被复制,这样效率很低。我们可以用引用来避免这种情况。
Set <span style="color:#0000ff;">operator</span> *( Set &Set1,Set &Set2)
{ Set res;
<span style="color:#0000ff;">for</span>(<span style="color:#0000ff;">int</span> i=<span style="color:#800080;">0</span>;i<Set1.card;++i)
<span style="color:#0000ff;">for</span>(<span style="color:#0000ff;">int</span> j=<span style="color:#800080;">0</span>;j>Set2.card;++j)
<span style="color:#0000ff;">if</span>(Set1.elems[i]==Set2.elems[j])
{
res.elems[res.card++]=Set1.elems[i];
<span style="color:#0000ff;">break</span>;
}
<span style="color:#0000ff;">return</span> res;
}

三、引用返回值

如果一个函数返回了引用,那么该函数的调用也可以被赋值。这里有一函数,它拥有两个引用参数并返回一个双精度数的引用:
<span style="color:#0000ff;">double</span> &max(<span style="color:#0000ff;">double</span> &d1,<span style="color:#0000ff;">double</span> &d2)
{
<span style="color:#0000ff;">return</span> d1>d2?d1:d2;
}
由于max()函数返回一个对双精度数的引用,那么我们就可以用max() 来对其中较大的双精度数加1:
max(x,y)+=<span style="color:#800080;">1.0</span>;
复制代码



 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
值传递指针传递引用传递是函数传递参数的三种方式,它们的区别在于传递方式不同,下面分别给出三种传递方式的代码和解释: 1. 值传递 值传递是指将参数的值复制一份给函数参数,函数内部对参数的修改不会影响到函数外部的值。 ``` #include <stdio.h> void swap(int a, int b) { int temp = a; a = b; b = temp; } int main() { int x = 10, y = 20; swap(x, y); printf("x = %d, y = %d\n", x, y); return 0; } ``` 上面的代码中,swap函数采用值传递的方式传递参数,对参数进行交换。在main函数中,调用swap函数交换x和y的值。但是由于swap函数中的参数是值传递的,因此对参数的修改不会影响到main函数中的x和y变量。因此,最终输出的结果是x = 10, y = 20。 2. 指针传递 指针传递是指将参数的地址传递给函数,函数内部可以通过指针来访问和修改参数的值。函数内部对指针所指向的值的修改会影响到函数外部的值。 ``` #include <stdio.h> void swap(int *a, int *b) { int temp = *a; *a = *b; *b = temp; } int main() { int x = 10, y = 20; swap(&x, &y); printf("x = %d, y = %d\n", x, y); return 0; } ``` 上面的代码中,swap函数采用指针传递的方式传递参数,对参数进行交换。在main函数中,调用swap函数交换x和y的值。由于swap函数中的参数是指针类型,因此可以通过指针来访问和修改参数的值。因此,最终输出的结果是x = 20, y = 10。 3. 引用传递 引用传递是指将参数的引用(别名)传递给函数,函数内部可以通过引用来访问和修改参数的值。函数内部对引用所指向的值的修改会影响到函数外部的值。 ``` #include <stdio.h> void swap(int &a, int &b) { int temp = a; a = b; b = temp; } int main() { int x = 10, y = 20; swap(x, y); printf("x = %d, y = %d\n", x, y); return 0; } ``` 上面的代码中,swap函数采用引用传递的方式传递参数,对参数进行交换。在main函数中,调用swap函数交换x和y的值。由于swap函数中的参数是引用类型,因此可以通过引用来访问和修改参数的值。因此,最终输出的结果是x = 20, y = 10。 总的来说,值传递指针传递引用传递是函数传递参数的三种方式,它们的区别在于传递方式不同。值传递是将参数的值复制一份给函数参数,函数内部对参数的修改不会影响到函数外部的值;指针传递是将参数的地址传递给函数,函数内部可以通过指针来访问和修改参数的值,函数内部对指针所指向的值的修改会影响到函数外部的值;引用传递是将参数的引用(别名)传递给函数,函数内部可以通过引用来访问和修改参数的值,函数内部对引用所指向的值的修改会影响到函数外部的值。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值