引用传递和值传递,从上学那会儿就开始强调的概念,不管你是计算机相关专业还是自学Java,一定听过这么一句话:
方法调用参数如果是对象,那就是引用传递,如果是基本数据类型,就是值传递。
比如:function(Object o)就是引用传递,function(int i)就是值传递。这两个概念似乎很好理解,我们只需要记住对象和基本数据类型的区别就行了。但是,真的是这样吗?
【资料图】
有一段代码如下:
public static void main(String[] args) { int i = 0; System.out.println(i); change(i); System.out.println(i); } private static void change(int i) { i = 1; }
输出结果比较好猜测,也应该都能答对:
00
下一个问题,如果将int改成String呢?
public static void main(String[] args) { String s = "0"; System.out.println(s); change(s); System.out.println(s); } private static void change(String s) { s = "1"; }
输出的结果如下:
00
嗯?有疑问了吧?不是引用传递吗?我在方法里命名修改了s的值,为什么输出还是”0“呢?难度String作为Object有什么特殊性?
别急,继续看下一段代码:
public static void main(String[] args) { Person p = new Person("0"); System.out.println(p); change(p); System.out.println(p); } private static void change(Person p) { p = new Person("1"); } static class Person{ String name; public Person(String name) { this.name = name; } @Override public String toString() { return "Person{" + "name="" + name + "\"" + "}"; } }
结果会输出什么?
Person{name="0"}Person{name="0"}
看来String和其他Object没什么不同,可是这样的结果好像不太符合我们对引用传递的认知啊。其实我感觉这两个概念没有必要区分,实质是一回事,都是将栈中引用复制了一份传递到方法中,无论在方法中如何对引用操作,都是操作的副本,只是对于基本数据类型来说,值存储在栈中,引用存储的就是值,而对象来说,引用中存储的是对象在堆中的内存地址,参数传递时生成的副本仍然指向了原来引用指向的对象,所以如果直接操作该对象是有效的。简单画个图方便理解:
如果对p的操作不是将该引用指向一个新的值,而是对p指向的对象进行操作,就能看到所谓引用传递的效果了例如:
public static void main(String[] args) { Person p = new Person("0"); System.out.println(p); change(p); System.out.println(p); } private static void change(Person p) { p.name = "1"; } static class Person{ String name; public Person(String name) { this.name = name; } @Override public String toString() { return "Person{" + "name="" + name + "\"" + "}"; } }
此时,执行结果为:
Person{name="0"}Person{name="1"}
总结:
Java进行方法调用时参数传递是将栈中的引用复制了一份到该方法的工作区,如果引用指向了一个堆中的对象,那么副本也指向这个对象。
标签: