在 Java 中,调用方法时,参数的传递分为两种类型:
- 传递值
- 传引用值
如果是传递值类型,意味着方法接收的是实际值的一个副本,原始值不会被改变。例如 :
public class Test {
public static void main(String[] args) throws Exception {
int i = 10;
System.out.println("原始值 i =" + i);
add(i);
System.out.println("调用 add() 后的 i = " + i);
}
private static int add(int i) {
i = i + 10;
System.out.println("add() 中的 i = " + i);
return i;
}
}
执行结果:
调用 add()
方法后,值 i
并没有发生改变,还是 10。为什么呢?因为 main()
调用 add()
传递的是 i
的一个副本,我们可以简单理解为它是另外一个变量 j
,在 add()
中改变的是这个 j
而不是原始 i
。
如果是传引用,那么它传递的是对象的引用值,我们是可以通过这个引用值来改变对象的状态,例如:
public class Test {
public static void main(String[] args) throws Exception {
int[] nums = {1,2,3,4};
System.out.println("add() 前的nums : "+ Arrays.toString(nums));
add(nums);
System.out.println("add() 后的nums : "+ Arrays.toString(nums));
}
private static void add(int[] nums) {
for (int i = 0 ; i < nums.length ; i++) {
nums[i] += 1;
}
System.out.println("add()中的nums : "+ Arrays.toString(nums));
}
}
执行结果:
在 Java中,除了基本数据类型(如 int
, float
, char
, boolean
等)之外,几乎所有使用 new
关键字创建的实例或者由特定方法返回的类型都是对象引用。例如:
- 类实例:任何通过
new
关键字创建的类的实例都是对象引用。例如,String str = new String("Hello");
中的str
是一个指向String
对象的引用。 - 数组:所有的数组,无论是基本数据类型的数组还是对象数组,都是对象引用。例如,
int[] arr = new int[10];
中的arr
是一个指向整型数组的引用。 - 集合类:Java中的集合类(如
ArrayList
,HashMap
,HashSet
等)都是对象引用。 - 接口类型:接口本身不能被实例化,但可以引用实现了该接口的类的对象。例如,
List<String> list = new ArrayList<>();
中的list
是一个指向ArrayList
实例的引用。 - 字符串:虽然字符串可以直接赋值(如
String s = "Hello";
),但字符串在Java中实际上是对象,因此s
是一个指向String
对象的引用。 - 包装类:基本数据类型的包装类(如
Integer
,Character
,Double
等)也是对象引用。
对于传引用值,我们一定要注意。Java 传递的是引用值,而不是引用本身。当我们传递对象时,传递的是对象引用的值
,是引用地址的副本,而不是对象本身。**我们可以通过这个引用来改变对象的状态,但不能改变引用本身所指向的对象。**这句话比较难理解,我们分拆开来看:
- 可以通过这个引用来改变对象的状态
当我们有一个对象引用时,我们可以使用这个引用来访问和修改这个对象的属性或者调用对象的方法,从而改变对象的状态。例如,我们可以用ArrayList
对象的引用,通过调用 add()
来添加元素,从而改变这个 ArrayList
对象的内容。
public class Test {
public static void main(String[] args) throws Exception {
List<String> list = new ArrayList<>(Arrays.asList("死磕 Java","死磕 Java 新特性","Java 面试 600 讲"));
System.out.println("add() 前的 list : "+ list);
add(list);
System.out.println("add() 后的 list: "+ list);
}
private static void add(List<String> list) {
list.add("死磕 Netty");
}
}
运行结果
- 不能改变引用本身所指向的对象
当我们将引用作为参数传递给方法时,我们传递的是这个引用的副本值,我们不能改变该引用所指向的原始对象。换句话说,就是不能使引用指向一个完全不同的对象。例如我们上面的方法,将 List 重新赋值一个新对象:
public class Test {
public static void main(String[] args) throws Exception {
List<String> list = new ArrayList<>(Arrays.asList("死磕 Java","死磕 Java 新特性","Java 面试 600 讲"));
System.out.println("add() 前的 list : "+ list);
add(list);
System.out.println("add() 后的 list: "+ list);
}
private static void add(List<String> list) {
list = new ArrayList<>();
}
}
执行结果:
如果传递的是引用的话,那么add() 后的 list 应该为空。
所以,在 Java 中,无论是基本类型还是对象,Java总是采用传值的方式。对于基本类型,传递的是实际值的副本;对于对象,传递的是对象引用地址的副本。