一、前言
Java是值传递的,对基本型变量而言的,传递的是该变量的一个副本,改变副本不影响原变量。对于对象型变量而言的,传递的是该对象地址的一个副本,,并不是原对象本身 ,这里也有人说是引用传递。由于副本的地址和原对象地址一致,因此对副本的值进行操作时,会同步改变原对象值。
但是一旦副本的地址被改变,副本的值的操作则不会影响原对象地址。(重点)
二、常见例子
1、基本类型参数的值传递
public class Test { public static void main(String[] args) { int num = 0 ; changeNum(num); System.out.println("num="+num); } private static void changeNum(int num) { num = 1; } }
最终输出结果是:num=0
因为这里 changeNum(num);语句中的num传递的是num的副本,也就是形参,所以当副本变为1,对实参原对象不影响,原对象还是num=0;
2、封装类型参数
public class Test { public static void main(String[] args) { Product p = new Product(); p.setProName("before"); p.setNum(0); changeProduct(p); System.out.println("p.proName="+p.getProName()); System.out.println("p.num="+p.getNum()); } private static void changeProduct(Product p) { p.setProName("after"); p.setNum(1); } } class Product { private int num; private String proName; public int getNum() { return num; } public void setNum(int num) { this.num = num; } public String getProName() { return proName; } public void setProName(String proName) { this.proName = proName; } }
最终输出结果是:p.proName=after
和p.num=1
。
这里我们可以看到 changeProduct(p); 该语句传递的是p内存中存储的地址的副本,该地址就是new Product();的地址。在changeProduct方法中对p地址的副本指向的值进行操作,由于副本地址和原地址一样,所以相当于最终p地址指向的值也会发生变化。所以这里的changeProduct方法能改变实参。
但是存在以下特殊情况,就是前言说的一旦副本的地址被改变。我们再 changeProduct() 方法中添加一条语句;
private static void changeProduct(Product p) { p = new Product();//添加这条语句,相当于P的地址变成new Product()新对象的地址,不是主函数中的对象的地址 p.setProName("after"); p.setNum(1); }
这里添加的这条语句 改变了副本的地址,导致副本指向新对象的地址,因此变成对新对象的操作,不影响主函数中的原对象,最终输出结果是:p.proName=before
和p.num=0
。
3、容易忽略的特殊类型String
首先要清楚一点,这种说法不正确:String str = "java",这就相当于String str = new String("java"),这是创建一个新的String的过程,不单单是赋值的过程。
public class Test { public static void main(String[] args) { String str = "ab"; changeString(str); System.out.println("str="+str); } private static void changeString(String str) { str = "cd"; } }
猜猜这里最终结果是什么?
最终输出结果是:str=ab 。(没猜对的说明上面第二点还没理解清楚)
解析:
str1 = "cd"
str2 = new String("cd");
System.out.println(str1 == str2); // false
说明str = "cd";不能被解释为如下:str = new String("cd");
这里之所以最终显示结果是“ad”,是因为main中的str的地址是常量池“ad”的地址,而在str = "cd"说明此时str的地址变成常量池“cd”的地址,地址改变,对实参原来的str无影响,所以最终输出的还是“ad”.
4、常见面试题
public class Example { String str = new String("good"); char[] ch = { 'a', 'b', 'c' }; public static void main(String args[]) { Example ex = new Example(); ex.change(ex.str, ex.ch); System.out.print(ex.str + " and "); System.out.print(ex.ch); } public void change(String str, char ch[]) { str = "test ok"; ch[0] = 'g'; } }
从如下4个选项选出最终输出结果
A 、 good and abc B 、 good and gbc C 、 test ok and abc D 、 test ok and gbc
正确答案: B
解析:
ex.str如第3点所说的,其传递的地址副本在change方法中被改变,因此不影响原地址的对象,所以ex.str不变;
ch数组是对象,数组的父类也是Object。传递的是数组地址的副本,因此在change中副本地址没改变,相当于对原对象进行操作,所以ch数组的值发送变化。