你真的真的理解Java的按引用传递吗?

简介:        今天在博客上看到《你真的理解Java的按引用传递吗?》这篇博文,就好奇进去看了一下,结果发现,其实说了半天,并没有特别清楚的解释。尤其是对于传递String类型时的例子时,有点发蒙。

       今天在博客上看到《你真的理解Java的按引用传递吗?》这篇博文,就好奇进去看了一下,结果发现,其实说了半天,并没有特别清楚的解释。尤其是对于传递String类型时的例子时,有点发蒙。

       接触Java也有好几年了,本来以为这些简单自己早就懂了,结果在看到最后一个例子时,直接就把答案给猜错了。

public class Test5{
    public static void main(String[] args) {
        Object o = new Object();
        System.out.println(o);
        change(o);
        System.out.println(o);
    }
    public static void change(Object o){
        o = null;
    }
}
       上面这个demo的结果是:
java.lang.Object@2a139a55
java.lang.Object@2a139a55
       而不是
java.lang.Object@2a139a55
null

       仔细研究了一下,现在终于是弄懂了。原来以前都是自以为懂了。

       先不解析上面这个demo,我们拿下面这个来说明:

public class AAA {
	private String str = "123";
	public static void main(String[] args) {
		AAA aaa = new AAA();
		System.out.println(aaa.str);
		aaa.change(aaa);
		System.out.println(aaa.str);
		System.out.println("-------------------------------");
		AAA bbb = new AAA();
		System.out.println(bbb.str);
		bbb.change2(bbb);
		System.out.println(bbb.str);
	}
	public void change(AAA a) {
		a.str = "abc";
	}
	public void change2(AAA a) {
		a = new AAA();
		a.str = "abc";
	}
}
       答案是什么呢:


       为什么不一样,按照在书上看到的——“java传递的是引用,也就是对象的地址”,那change方法把源对象的str属性给改了,为什么change2不能改呢?


       第一副图,对应对象aaa,第二副图对应对象bbb。

       第一副:首先new了一个对象aaa,里面有一个str,指向字符串“123”的地址。然后调用change(aaa)后,由于java方法传递的是地址值的拷贝,所以参数a对象也是指向了aaa原来的堆空间。然后更改str,把参数a中str的指向了“abc”所在空间,即把aaa的str的地址给换成了“abc”,所以最后在输出的时候就会输出“abc”了。

       第二副:首先new了一个对象bbb,里面有一个str,指向字符串“123”的地址。然后调用change(bbb)后,由于java方法传递的是地址值的拷贝,所以参数a对象也是指向了aaa原来的堆空间。但是,方法里调用了b = new AAA();,这样就把b指向的地址换成了一个新对象的地址,同时里面有一个str,由于也是“123”,所以指向的是同一个“123”所属的地址。然后更改str,只是把参数对象a的str的地址给换成了“abc”(双线粗箭头),但是bbb对象中的str仍然没有变化,所以最后在输出的时候还是输出“123”了。


       所以说在传递对象类型时,就看在方法里参数的地址在修改其他属性前有没有发生改变:

    • 如果形参的(引用)地址发生了改变,那对原对象就没有影响;
    • 如果形参(引用)地址未改变,但属性有改变,原对象的属性会有随之改变;
    • 如果属性先改变,形参(引用)地址后改变,那原对象的属性在地址改变前有修改的会随之改变。

       所以我们会经常看到下面的这样的例子:

	public static void main(String[] args) {
		BBB bbb = new BBB();
		List<Object> list = new ArrayList<Object>();
		System.out.println(list.size());
		bbb.change(list);
		System.out.println(list.size());
	}
	public void change(List<Object> list) {
		for (int i = 0; i < 3; i++) {
			list.add("00"+i);
		}
	}
       最后输出list.size=3,因为list的地址没有发生变化,所以方法中对list增删对象,原对象就会有变化。


       所以现在再看传递String的这个方法,是不是就简单多了。因为在方法中参数的地址改变了,所以就不会影响到源对象了。所以还是输出“123”。

public class Test2{
    public static void main(String[] args) {
        String str = "123";
        System.out.println(str);
        change(str);
        System.out.println(str);
    }
    public static void change(String str){
        str = "abc";
    }
}

       那再回头看看文章里第一个例子是不是就也清楚了。

目录
相关文章
|
安全 Java
【面试题精讲】为什么 Java 不引入引用传递呢?
【面试题精讲】为什么 Java 不引入引用传递呢?
|
5月前
|
Java
java是值传递还是引用传递
本文澄清了Java中参数传递的常见误解,总结出Java采用“值传递”的方式。对于基本类型,传递其值的拷贝,方法内修改不影响原值;而对于对象类型,则传递其引用地址的拷贝,尽管是拷贝,但因指向同一对象,故方法内的修改会影响原对象状态。形参仅在方法内部有效,而实参则是调用方法时传递的具体值。通过示例和比喻(如复刻仓库钥匙),形象地解释了值传递、引用传递及Java特有的“共享对象传递”概念,帮助理解不同情况下参数传递的行为差异。
|
5月前
|
Java
java中的值传递和引用传递
【8月更文挑战第3天】在Java中,值传递用于基本数据类型,传递的是值的副本,因此方法内的修改不影响原值;而引用传递用于对象和数组,虽传递的是引用的副本,但对对象内容的修改会影响原始对象。理解这两者对于正确处理方法调用及参数至关重要。
|
5月前
|
Java
java中的值传递和引用传递
【8月更文挑战第2天】在Java中,基本数据类型如`int`、`double`等采用值传递,传递的是变量值的副本,因此方法内的修改不影响原变量。对象类型则通过引用传递,传递的是对象引用的副本,允许方法内修改原对象。例如,对`StringBuilder`对象的修改会影响原始对象。
|
7月前
|
Java
Java的值传递与“引用传递”辨析
Java的值传递与“引用传递”辨析
33 0
|
8月前
|
JavaScript 前端开发 Java
【JAVA面试题】什么是引用传递?什么是值传递?
【JAVA面试题】什么是引用传递?什么是值传递?
|
8月前
|
Java
每日一道Java面试题:Java是值传递还是引用传递?
每日一道Java面试题:Java是值传递还是引用传递?
43 1
|
存储 缓存 算法
java引用传递引发的一系列问题
最近在学习java引用传递,其实引用传递就是和内存分配有关;
146 1
|
Java
JAVA参数传值机制中值传递和引用传递
JAVA参数传值机制中值传递和引用传递
104 0
【java筑基】——为什么java中方法的参数传递机制没有引用传递
【java筑基】——为什么java中方法的参数传递机制没有引用传递