本文转载自 https://blog.csdn.net/SEU_Calvin/article/details/70089977
1. 你觉得下面程序会输出什么
- public static void change(String s) {
- s = “123”;
- }
- public static void main(String args[]) {
- String s = “abc”;
- change(s);
- System.out.println(s);
- }
面试官写下了这段代码,问我s会输出什么。
我想起了各种排序算法直接把一个int数组array,通过参数传递给一个方法,在方法里完成数据的移动后,直接在main()函数中输出array[]中的值就是修改过的。所以我回答说,那应该输出123吧。
然后面试官说,不对,是输出abc,你觉得原因是什么。
我想了想回答说,String是final的,所以值是不变的。
接着面试官写下了如下代码:
- public static void main(String[] args) {
- MyClass myClass = new MyClass();
- change(myClass);
- System.out.println(myClass.val);
- }
- private static void change(MyClass myClass) {
- myClass = new MyClass();
- myClass.val = 2;
- }
- public static class MyClass{
- int val = 1;
- }
问我MyClass类不是final的,但为什么这里还是会输出1。我有点被问懵了,其实后来面试结果又仔细看一下,是自己忽略掉了第八行代码,其实把第八行代码注释掉,肯定就输出2了。
2. Java参数传值
2.1 值传递
先从一个例子说起:
- public static void change(int i, int j) {
- inttemp = i;
- i =j;
- j =temp;
- }
- public static void main(String[] args) {
- inta = 3;
- intb = 4;
- change(a,b);
- System.out.println(a);
- System.out.println(b);
- }
输出为:
- 3
- 4
值传递是指方法调用时,实际参数把它的值传递给对应方法的形参,如change()方法中i和j,也在内存空间中分配了存储单元,这样形参在change()方法中的改变不会影响实际参数的值。
值传递的数据类型包括,八种基本数据类型和String。
2.2 引用传递
值传递的数据类型包括,八种基本数据类型和String,八种数据类型还比较好理解,比如2.1中的int,但是String并不是基本数据类型,这要怎么理解呢。比如下面这个例子。
- public class Example {
- String str = new String("abc");
- char[] ch = { 'a', 'b', 'c' };
- public static void main(String args[]) {
- Example ex = new Example();
- ex.change(ex.str, ex.ch);
- System.out.println(ex.str);
- System.out.println(ex.ch);
- }
- public void change(String str, char ch[]) {
- //ch = new char[]{'a','b','c'};
- str = "change";
- ch[0] = 'c';
- }
- }
这个例子的输出为:
- abc
- cbc
要知道String和char数组都是引用类型,不是基本类型。
当我们把str作为参数传入方法后,会新建另一个变量,已经不是原来的变量了,但是他们指向的数据区域都一样,所以如果你在方法中改变了str指向的数据区域,即执行str = "change",那也只是改变新建的另一个变量所指向的数据区域,即指向一个新对象"change",str仍然指向原来的数据区域。所以会输出abc。
但是对于char数组的例子呢,即对象类型,也就是Object的子类(除了String),是把ch的引用传递进来,即引用传递。这样里面和外面的ch都指向了相同的数据区域,执行ch[0] = 'c',就会把这个数据区域里的第一个字符改成c,并没有改变内部ch的数据地址,所以这个修改也会反映到外部的ch。所以会输出cbc。如果你改变了内部ch所指向的数据区域,即把上面代码中注释的那一行打开,即执行ch = new char[]{'a','b','c'},这样ch已经指向一个新的数据区域。输出的结果肯定也就是abc了。
这样在1中面试官给的两段代码也就很容易解释了。
3 两种数据传递总结
值传递和引用传递的本质区别在于是否在传递的时候进行对象的内存拷贝。
基本类型是由于在JVM中存储区域不同于普通对象所以传递前会拷贝,传递的是拷贝后的值,但是对象在传递的时候不拷贝,直接传“引用值”,指向同一片对象堆内存区域,当然要注意String这种特殊情况。