Java函数参数传递

到底是传值 or  传引用(传地址)

结论:

1、基本数据类型作为参数传递时,是值得拷贝,无论你是怎么修改这个拷贝的,原来的值是不是被改变的。

2、对象作为参数传递时,其实是把对象在堆内存的中的地址拷贝了一份传给了参数。

一、基本类型作为参数传递

测试1:

结果:

所以可以看出,基本类型作为参数传递时,是传递值得拷贝,无论你是怎么修改这个拷贝,原值是不会改变的。

二、对象作为参数传传递

测试2:

结果:

可以看到值被修改了,那么是不是意味着传递的不是值得拷贝,而是值得本身?

其实不然的,我们可以再测试第三个例子

测试3:

结果:

可以看出居然sb的值不变,这就可以推断出,我们传递的对象其实并不是对象本身,而是一个地址的拷贝,而是类似了C里面的指针,多个指针同样指向了一块内存,就可以对其进行操作了。

我们以一个图来看看

StringBuffer sb = new StringBuffer("Hello ");

执行完后,就会在堆内存中开辟一个存储sb对象的地址,如图:

sb 是一个引用,里面存放的是一个地址 "@3a"

然后执行

changeData(sb);

就把sb传递给了changeData方法中的StringBuffer strBuf 。由于sb存放的是地址,所以strBuf也将存放相同的地址,如图:

 

 此时的,sb和strBuf由于存放的内存地址相同,都指向了 "Hello" 

strBuf.append("World!");

 执行changeData方法中的这一句后,改变了strBuf指向的内存中的值,如下图3所示

所以在测试2里面会输出

修改后sb = Hello World!

但是在测试3里面又做了一步

strBuf = new StringBuffer("World");

这一步相当于在内存中另外开辟的一个StrBuf的存储空间,strBuf由原来指向 "Hello",有重新的指向了 “liuzeyu12a”(“Hi”)

所以当执行完

 strBuf.append("World!");

通过上面的比较,就会很清楚的发现,由于sb 和strBuf中存放的地址不一样了。所以当strBuf指向

的内存值发生改变了,但sb 指向的内存值并不会改变,因此测试3输出了

修改前sb=Hello
修改后sb=Hello

--------------------------------

综上所述,我们就会明白,Java中对象作为参数传递时,是把对象在内存中的地址拷贝了一份传给了参数。

参考资料:https://www.cnblogs.com/hpyg/p/8005599.html

posted @ 2019-01-31 12:36  四季列车  阅读(367)  评论(0编辑  收藏  举报