java中的经典问题:传值与传引用


参数传递的秘密 

知道方法参数如何传递吗? 
记得刚开始学编程那会儿,老师教导,所谓参数,有形式参数和实际参数之分,参数列表中写的那些东西都叫形式参数,在实际调用的时候,它们会被实际参数所替代。 
编译程序不可能知道每次调用的实际参数都是什么,于是写编译器的高手就出个办法,让实际参数按照一定顺序放到一个大家都可以找得到的地方,以此作为方法调用的一种约定。所谓“没有规矩,不成方圆”,有了这个规矩,大家协作起来就容易多了。这个公共数据区,现在编译器的选择通常是“栈”,而所谓的顺序就是形式参数声明的顺序。 
显然,程序运行的过程中,作为实际参数的变量可能遍布于内存的各个位置,而并不一定要老老实实的呆在栈里。为了守“规矩”,程序只好将变量复制一份到栈中,也就是通常所说的将参数压入栈中。 
打起精神,谜底就要揭晓了。 
我刚才说什么来着?将变量复制一份到栈中,没错,“复制”! 

这就是所谓的值传递。


C语言的旷世经典《The C Programming Language》开篇的第一章中,谈到实际参数时说,“在C中,所有函数的实际参数都是传‘值’的”。 
马上会有人站出来,“错了,还有传地址,比如以指针传递就是传地址”。 
不错,传指针就是传地址。在把指针视为地址的时候,是否考虑过这样一个问题,它也是一个变量。前面的讨论中说过了,参数传递必须要把参数压入栈中,作为地址的指针也不例外。所以,必须把这个指针也复制一份。函数中对于指针操作实际上是对于这个指针副本的操作。 
Java的reference等于C的指针。所以,在Java的方法调用中,reference也要复制一份压入堆栈。在方法中对reference的操作就是对这个reference副本的操作。 
谜底揭晓 
好,让我们回到最初的问题上。 
在changeReference中对于reference的赋值实际上是对这个reference的副本进行赋值,而对于reference的本尊没有产生丝毫的影响。 
回到调用点,本尊醒来,它并不知道自己睡去的这段时间内发生过什么,所以只好当作什么都没发生过一般。就这样,副本消失了,在方法中对它的修改也就烟消云散了。 

也许你会问出这样的问题,“听了你的解释,我反而对changeInteger感到迷惑了,既然是对于副本的操作,为什么changeInteger可以运作正常?” 
呵呵,很有趣的大脑短路现象。 
好,那我就用前面的说法解释一下changeInteger的运作。 
所谓复制,其结果必然是副本完全等同于本尊。reference复制的结果必然是两个reference指向同一块内存空间。 
虽然在方法中对于副本的操作并不会影响到本尊,但对内存空间的修改确实实实在在的。 
回到调用点,虽然本尊依然不知道曾经发生过的一切,但它按照原来的方式访问内存的时候,取到的确是经过方法修改之后的内容。 
于是方法可以把自己的影响扩展到方法之外。


1.所有的参数传递都是 传值,从来没有 传引用 这个事实;
2. 所有的参数传递都会在 程序运行栈上 新分配一个 值 的复制品;

3.java只有按值传递,所谓的按地址(引用)传递,也属于按值传递,只不过这个“值”是个地址;

4.对于引用类型的传参也是传值的,传的是引用类型的值,其实就是对象的地址;

1.java参数是传递值的。
2.java所有对像变量都是对像的引用;

5.或者说:传递过去的都是拷贝,区别在于拷贝的是基本数据类型还是引用;

6.函数的形式参数,是传入参数的拷贝;引用变量之间拷贝的是【地址】,基本变量之间拷贝的是 内存中的值 (被称为直接量);
7.对象本身,与对象的地址 是2个东西,函数之间如果想【传递对象】,只能通过传递对象的地址来实现&#

阅读终点,创作起航,您可以撰写心得或摘录文章要点写篇博文。去创作
  • 13
    点赞
  • 65
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 5
    评论
1. 简单类型是按递的   Java 方法的参数是简单类型的时候,是按递的 (pass by value)。这一点我们可以通过一个简单的例子来说明: /* 例 1 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void test(boolean test) { test = ! test; System.out.println("In test(boolean) : test = " + test); } public static void main(String[] args) { boolean test = true; System.out.println("Before test(boolean) : test = " + test); test(test); System.out.println("After test(boolean) : test = " + test); } } 运行结果: Before test(boolean) : test = true In test(boolean) : test = false After test(boolean) : test = true   不难看出,虽然在 test(boolean) 方法改变了进来的参数的,但对这个参数源变量本身并没有影响,即对 main(String[]) 方法里的 test 变量没有影响。那说明,参数类型是简单类型的时候,是按递的。以参数形式递简单类型的变量时,实际上是将参数的作了一个拷贝进方法函数的,那么在方法函数里再怎么改变其,其结果都是只改变了拷贝的,而不是源。   2. 什么是引用   Java还是引用问题主要出在对象的递上,因为 Java 简单类型没有引用。既然争论提到了引用这个东西,为了搞清楚这个问题,我们必须要知道引用是什么。   简单的说,引用其实就像是一个对象的名字或者别名 (alias),一个对象在内存会请求一块空间来保存数据,根据对象的大小,它可能需要占用的空间大小也不等。访问对象的时候,我们不会直接是访问对象在内存的数据,而是通过引用去访问。引用也是一种数据类型,我们可以把它想象为类似 C 语言指针的东西,它指示了对象在内存的地址——只不过我们不能够观察到这个地址究竟是什么。   如果我们定义了不止一个引用指向同一个对象,那么这些引用是不相同的,因为引用也是一种数据类型,需要一定的内存空间来保存。但是它们的是相同的,都指示同一个对象在内存的位置。比如 String a = "Hello"; String b = a;   这里,a 和 b 是不同的两个引用,我们使用了两个定义语句来定义它们。但它们的是一样的,都指向同一个对象 "Hello"。也许你还觉得不够直观,因为 String 对象的本身是不可更改的 (像 b = "World"; b = a; 这种情况不是改变了 "World" 这一对象的,而是改变了它的引用 b 的使之指向了另一个 String 对象 a)。那么我们用 StringBuffer 来举一个例子: /* 例 2 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void main(String[] args) { StringBuffer a = new StringBuffer("Hello"); StringBuffer b = a; b.append(", World"); System.out.println("a is " + a); } } 运行结果: a is Hello, World   这个例子 a 和 b 都是引用,当改变了 b 指示的对象的的时候,从输出结果来看,a 所指示的对象的也改变了。所以,a 和 b 都指向同一个对象即包含 "Hello" 的一个 StringBuffer 对象。   这里我描述了两个要点: 引用是一种数据类型,保存了对象在内存的地址,这种类型即不是我们平时所说的简单数据类型也不是类实例(对象); 不同的引用可能指向同一个对象,换句话说,一个对象可以有多个引用,即该类类型的变量。   3. 对象是如何递的呢   关于对象的递,有两种说法,即“它是按递的”和“它是按引用递的”。这两种说法各有各的道理,但是它们都没有从本质上去分析,即致于产生了争论。   既然现在我们已经知道了引用是什么东西,那么现在不妨来分析一下对象作是参数是如何递的。还是先以一个程序为例: /* 例 3 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void test(StringBuffer str) { str.append(", World!"); } public static void main(String[] args) { StringBuffer string = new StringBuffer("Hello"); test(string); System.out.println(string); } } 运行结果:   Hello, World!   test(string) 调用了 test(StringBuffer) 方法,并将 string 作为参数递了进去。这里 string 是一个引用,这一点是勿庸置疑的。前面提到,引用是一种数据类型,而且不是对象,所以它不可能按引用递,所以它是按递的,它么它的究竟是什么呢?是对象的地址。   由此可见,对象作为参数的时候是按递的,对吗?错!为什么错,让我们看另一个例子: /* 例 4 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void test(String str) { str = "World"; } public static void main(String[] args) { String string = "Hello"; test(string); System.out.println(string); } }   运行结果: Hello   为什么会这样呢?因为参数 str 是一个引用,而且它与 string 是不同的引用,虽然它们都是同一个对象的引用。str = "World" 则改变了 str 的,使之指向了另一个对象,然而 str 指向的对象改变了,但它并没有对 "Hello" 造成任何影响,而且由于 string 和 str 是不同的引用,str 的改变也没有对 string 造成任何影响,结果就如例所示。   其结果是推翻了参数按递的说法。那么,对象作为参数的时候是按引用递的了?也错!因为上一个例子的确能够说明它是按递的。   结果,就像光到底是波还是粒子的问题一样,Java 方法的参数是按什么递的问题,其答案就只能是:即是按递也是按引用递,只是参照物不同,结果也就不同。   4. 正确看待还是引用问题   要正确的看待这个问题必须要搞清楚为什么会有这样一个问题。   实际上,问题来源于 C,而不是 Java。   C 语言有一种数据类型叫做指针,于是将一个数据作为参数递给某个函数的时候,就有两种方式:,或是指针,它们的区别,可以用一个简单的例子说明: /* 例 5 */ /** * @(#) test.c * @author fancy */ void SwapValue(int a, int b) { int t = a; a = b; b = t; } void SwapPointer(int * a, int * b) { int t = * a; * a = * b; * b = t; } void main() { int a = 0, b = 1; printf("1 : a = %d, b = %d/n", a, b); SwapValue(a, b); printf("2 : a = %d, b = %d/n", a, b); SwapPointer(&a, &b); printf("3 : a = %d, b = %d/n", a, b); } 运行结果: 1 : a = 0, b = 1 2 : a = 0, b = 1 3 : a = 1, b = 0   大家可以明显的看到,按指针递参数可以方便的修改通过参数递进来的,而按递就不行。   当 Java 成长起来的时候,许多的 C 程序员开始转向学习 Java,他们发现,使用类似 SwapValue 的方法仍然不能改变通过参数递进来的简单数据类型的,但是如果是一个对象,则可能将其成员随意更改。于是他们觉得这很像是 C 语言/指针的问题。但是 Java 没有指针,那么这个问题就演变成了/引用问题。可惜将这个问题放在 Java 进行讨论并不恰当。   讨论这样一个问题的最终目的只是为了搞清楚何种情况才能在方法函数方便的更改参数的并使之长期有效。   Java ,改变参数的有两种情况,第一种,使用赋号“=”直接进行赋使其改变,如例 1 和例 4;第二种,对于某些对象的引用,通过一定途径对其成员数据进行改变,如例 3。对于第一种情况,其改变不会影响到方法该方法以外的数据,或者直接说源数据。而第二种方法,则相反,会影响到源数据——因为引用指示的对象没有变,对其成员数据进行改变则实质上是改变的该对象。   5. 如何实现类似 swap 的方法   还是引用问题,到此已经算是解决了,但是我们仍然不能解决这样一个问题:如果我有两个 int 型的变量 a 和 b,我想写一个方法来交换它们的,应该怎么办?   结论很让人失望——没有办法!因此,我们只能具体情况具体讨论,以经常使用交换方法的排序为例: /** 例 6 */ /** * @(#) Test.java * @author fancy */ public class Test { public static void swap(int[] data, int a, int b) { int t = data[a]; data[a] = data[b]; data[b] = t; } public static void main(String[] args) { int[] data = new int[10]; for (int i = 0; i < 10; i++) { data[i] = (int) (Math.random() * 100); System.out.print(" " + data[i]); } System.out.println(); for (int i = 0; i < 9; i++) { for (int j = i; j < 10; j++) { if (data[i] > data[j]) { swap(data, i, j); } } } for (int i = 0; i < 10; i++) { System.out.print(" " + data[i]); } System.out.println(); } } 运行结果(情况之一): 78 69 94 38 95 31 50 97 84 1 1 31 38 50 69 78 84 94 95 97   swap(int[] data, int a, int b) 方法在内部实际上是改变了 data 所指示的对象的成员数据,即上述讨论的第二种改变参数的方法。希望大家能够举一反三,使用类似的方法来解决相关问题

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

-江南听雨-

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值