第002弹:Java 中的值传递和引用传递

  在 Java 的代码开发过程中,为了尽可能提高方法的复用性,明确方法的作用,同时防止一个方法内部过于臃肿的问题,往往会创建许多方法,那么不可避免地会涉及到参数传递的问题。通常来说,我们将 Java 中的参数传递分为两种:值传递和引用传递。

  • 值传递:参数在进入方法时,将入参深度复制一个副本,在方法内部操作的是入参的副本,在方法执行完毕之后,外部的入参没有发生任何变化。
  • 引用传递:在方法内部操作的是参数本身,对入参做出的修改会保留到方法的外部。

  那么在 Java 中,哪些情况属于值传递,哪些情况属于引用传递呢?

1.      入参的类型

  有一种错误的见解被广为流传:如果入参是基本类型,属于值传递;如果入参不是基本类型,则属于引用传递。或者说,再深入探讨一点,和入参存储的位置相关。(基本类型存储在堆栈中,对象存储在堆中)

  以上这种说法其实并不完全正确。前半句基本认同,但是对于后半句,我可以很轻松地找到如下反例:

  由上图可知,字符串 String 类型产生的对象,是存储在堆中的非基本类型。根据以上看法,这种参数传递方式应该是引用传递,那么对字符串做出的修改应该会保存到 change(String) 方法之外,然而最终的输出结果并不是这样。把 String 改成 StringBuffer,做类似的操作,得出的结果也和 String 一致的。

  结论:基本类型的参数传递,一定是值传递;但是非基本类型的参数,其传递方式不一定是引用传递,需要进一步地分析。

2.      方法的返回类型

  另外有一种错误的看法,参数传递方式,和方法是否拥有返回值有关,如果一个方法有返回值,那么参数一定是按照值传递的。看一下如下的例子:

 1     public static void main(String[] args) {
 2         Person p1 = new Person();
 3         p1.setAge(20);
 4         p1.setGender(0);
 5         p1.setName("哈哈");
 6         Person p2 = new Person();
 7         BeanUtil.copySameFieldsObject(p1, p2);
 8         change1(p1);
 9         System.out.println(p1);
10         change2(p2);
11         System.out.println(p2);
12     }
13
14     private static void change1(Person p) {
15         p.setAge(30);
16         p.setGender(1);
17         p.setName("呵呵");
18     }
19
20     private static Person change2(Person p) {
21         p.setAge(30);
22         p.setGender(1);
23         p.setName("呵呵");
24         return null;
25     }

Test1

  如果说参数传递的方式,和方法的返回值有关,那么以上的两次输出结果一定是不同的(Person 类已经重写了 toString() 方法),因为根据以上推论,第一种方法是引用传递,第二种方法是值传递,但是实际上两次的输出结果是相同的。

  结论:参数传递方式,与方法是否有返回值,返回值的类型没有关系。

3.      真正决定入参传递方式的因素

  对于非基本类型的入参,其参数传递的方式是不定的。可以看一下如下例子:

 1     public static void main(String[] args) {
 2         Person p1 = new Person();
 3         p1.setAge(20);
 4         p1.setGender(0);
 5         p1.setName("哈哈");
 6         Person p2 = new Person();
 7         BeanUtil.copySameFieldsObject(p1, p2);
 8         Person p3 = new Person();
 9         BeanUtil.copySameFieldsObject(p1, p3);
10         change1(p1);
11         System.out.println(p1);
12         change2(p2);
13         System.out.println(p2);
14         change3(p3);
15         System.out.println(p3);
16     }
17
18     private static void change1(Person p) {
19         p.setAge(30);
20         p.setGender(1);
21         p.setName("呵呵");
22     }
23
24     private static void change2(Person p) {
25         p = new Person("呵呵", 1, 30);
26     }
27
28     private static void change3(Person p) {
29         p.setName("呵呵");
30         p = new Person("呵呵", 1, 30);
31     }

Test2

  以上程序中, BeanUtil.copySameFieldsObject() 方法的作用是深度复制一份第一个参数的内容,给第二个参数,输出结果如下图所示:

通过以上研究可以得出如下结论:

  • 可以认为非基本类型的入参的参数传递方式为引用传递,但是根据方法内部执行的代码,这种传递方式存在变数,可能被转化为值传递。
  • 可以认为,方法的入参是一个对象的引用,记为 p,存放在堆栈中,这个引用指向堆中的一片内存,记为 q。当整个方法只会修改 q 的内容,而 p 始终指向 q 时,可以把整个参数传递,作为引用传递来看待。
  • 当方法内部,如果代码企图将 p 指向另一片内存 t,这时,JVM 会创建另一个引用 r,让 r 指向 t,而 p 仍然指向 q。
  • 这种试图更改指针指向的行为,主要是创建一个新的对象。创建对象的具体方法,见上一章节的内容。还包括几种特殊形式,如使用操作符“=”,创建 String 类型的对象。另外需要额外注意的是某些方法,内部实现使用了 new 创建对象,如 String.concat(String) 方法。
  • 基本类型的8种包装类型,可以当做基本类型处理,其参数传递方式虽然是引用传递,但是可以认为与值传递等价。因为这8种包装类型和 String 类型相同,其内部的数据是不可变的,这意味着任何的变动,本质上是在内存中开辟了一个新的对象。
时间: 2024-08-07 08:38:44

第002弹:Java 中的值传递和引用传递的相关文章

JAVA中只有值传递

今天,我在一本面试书上看到了关于java的一个参数传递的问题: 写道 java中对象作为参数传递给一个方法,到底是值传递,还是引用传递? 我毫无疑问的回答:"引用传递!",并且还觉得自己对java的这一特性很是熟悉! 结果发现,我错了! 答案是: 值传递!Java中只有按值传递,没有按引用传递! 回家后我就迫不及待地查询了这个问题,觉得自己对java这么基础的问题都搞错实在太丢人! 综合网上的描述,我大概了解了是怎么回事,现在整理如下,如有不对之处望大神提出! 先来看一个作为程序员都熟

Java中的值传递和引用传递

解释 1.Java中有没有引用传递? 答:Java中只有按值传递,没有按引用传递! 2.当一个对象被当作参数传递到一个方法中后,在此方法中可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递? 答:是值传递. Java参数,不管是原始类型还是引用类型,传递的都是副本(有另外一种说法是传值,但是说传副本更好理解吧,传值通常是相对传址而言).如果参数类型是原始类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个跟之前所谈的传值是一样的,如果在函数中改变了副本

java中参数传递--值传递,引用传递

java中的参数传递--值传递.引用传递 参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递. 在 Java 应用程序中永远不会传递对象,而只传递对象引用.因此是按引用传递对象.Java 应用程序按引用传递对象这一事实并不意味着 Java 应用程序按引用传递参数.参数可以是对象引用,而 Java 应用程序是按值传递对象引用的. Java 应用程序中的变量可以为以下两种类型之一:引用类型或基本类型.当作为参数传递给一个方法时,处理这两种类型的方式是相同的.两

java中值传递和引用传递

本来今天刚学习的内容,然后去其他博客看了下,发现都吵起来了,就是名字原因,有的说java有值传递和引用传递,有的说引用传递本质就是值传递,我管你杂说的,只要自己理解好,代码知道运行结果就好了. 我用自己的话,自己的理解来解释下,反正都是自己写着玩,自己看的 值传递:传递的值,这个值以后怎么改变,源值不会发生改变的. 引用传递:将对象的引用地址传递过去,如果值发生改变,那么源值也发生改变. 代码如下: 值传递: public class Test1 { public static void mai

java中方法的参数传递机制(值传递还是引用传递)

看到一个java面试题: 问:当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?  答:是值传递.Java 编程语言只有值传递参数.当一个对象实例作为一个参数被传递到方法中时,参数的值就是该对象的引用一个副本.指向同一个对象,对象的内容可以在被调用的方法中改变,但对象的引用(不是引用的副本)是永远不会改变的. 以下是从其他文章里转的,只为加深理解 public class TempTest { private void te

Java中,函数参数的传递,是值传递还是引用传递

当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?  答:是值传递. 可以这么理解:有一个Person(name="S"),作为参数传给一个方法,在此方法中将此Person重新设置name=“B”,那么有一下结论 ①这个Person还是这个Person,一直都是他,但是他的名字被改了,不管是原来的引用还是方法里的引用,引用的都是一个对象,所以,所有引用的值都变了 ②如果在方法中重新new了一个Person赋给参数,

Java中的值传递与“引用传递”

首先,Java没有 引用传递 这么一说. Java只有值传递,传递的都是值,基本数据类型传递的是值,引用类型传递的是地址值. 我来理一下这其中犹如米线跟米面绞在一起了,再跟粉丝混在一起的关系. 好的,我们来看第一个例子: public static void main(String[] args) { int a = 1; int b = 4; System.out.println("Before : a=" + a + "," + "b=" +

java 中值传递和引用传递(转)

java中给函数传递参数的方式有两种:值传递和引用传递.一般而言,基本类型是值传递:引用类型是引用传递.但传值时到达发生了什么? 1.基本类型 8个基本类型(byte,short,int,long,float,double,char,boolean)是值传递. 1 public class ValueTest { 2 3 public static void main(String[] args) { 4 int a = 10; 5 changeVal(a); 6 System.out.prin

浅析java中的值传递与引用传递

public class Test { public static void main(String[] args) { String s = new String("aaa"); change(s); System.out.println(s); StringBuilder sb = new StringBuilder("111"); change(sb); System.out.println(sb); } static void change(String s