方法调用过程中,参数传递有两类:值传递、引用传递。
值传递,方法调用时,实际参数会把值传递给方法中对应的形式参数。在方法执行过程中,形式参数值的改变不会影响实际参数的值。
引用传递,也称地址传递。方法调用时,实际参数的引用(地址,而不是其真实值)传递给方法中对应形式参数。在方法执行过程中,针对形式参数的操作,实际上就是针对实际参数的操作,形式参数值的改变将会影响实际参数的值。
而在Java中,只有值传递。基本类型传递的是其值的副本,引用类型传递的是其引用的副本。
因此,将会衍生出来几种面试题,颇具迷惑性。
简单版:
public void print() {
int i = 0;
plus(i);
System.out.println(i);
}
public void plus(int i) {
i = i + 1;
}
此题答案为0,而并非是1。其原因就是plus方法接到的只是实际参数 i 的值的拷贝,即0。而plus方法内的 形式参数 i,plus方法无论对其做什么改变,都不影响原实际参数的值。
进阶版:
public void print() {
int i = 0;
plus(i);
System.out.println(i);
}
public int plus(int i) {
return i++;
}
public void print() {
int i = 0;
i = plus(i);
System.out.println(i);
}
public int plus(int i) {
return ++i;
}
此2题答案依然为0。传参部分解释同上。而迷惑你的,无非是plus方法多了返回值,还有,出现了 i++。一直以来,i++ 和 ++i 都是最易混淆的知识点,加上了这个知识点,更具迷惑性。为啥还是0呢?很简单,除了传参部分的原因,plus返回的值没有被接收/赋值,也是很重要的原因。
但是,plus方法返回的值被接收/赋值,结果就会是 1 吗?
public void print() {
int i = 0;
i = plus(i);
System.out.println(i);
}
public int plus(int i) {
return i++;
}
答案依然为0。为啥呢?简单的解释, i++ 先用后加!再来个变种:
public void print() {
int i = 0;
i = plus(i);
System.out.println(i);
}
public int plus(int i) {
return ++i;
}
这次答案终于为1了。为啥呢?简单的解释,++i 先加后用!
中级进阶版:
public void print() {
int a = 3;
int b = 4;
System.out.println("交换前:");
System.out.println("a:" + a + ",b:" + b);
swap(a,b);
System.out.println("交换后:");
System.out.println("a:" + a + ",b:" + b);
}
public void swap(int m, int n) {
int temp = m;
m = n;
n = temp;
}
答案是交换前后,并没有变化。原因依然是值传递,形式参数 m 和 n ,得到的依然是实际参数 a 和 b 的值副本,swap方法执行完毕后,m 和 n 作用域失效,实际参数 a 和 b 并无任何影响。
高级进阶版:
public void print() {
int ages[] = {1, 2, 3};
System.out.println("交换前:");
System.out.println("第一个元素:" + ages[0] + ",最后一个元素:" + ages[ages.length - 1]);
swap(ages);
System.out.println("交换后:");
System.out.println("第一个元素:" + ages[0] + ",最后一个元素:" + ages[ages.length - 1]);
}
public void swap(int[] ages) {
int temp = ages[0];
ages[0] = ages[ages.length - 1];
ages[ages.length - 1] = temp;
}
答案是互换成功。什么原因呢?值传递依然是传递的 ages 数组对象的引用的拷贝,均指向 ages 数组对象。所以,针对 ages 数组对象的引用的拷贝做操作,相当于直接对 ages 数组对象操作,均会改变其内容。
究极进阶版:
public void print() {
Person a = new Person(0);
Person b = new Person(100);
System.out.println("交换前:");
System.out.println("a的年龄:"+a.getAge()+"-----------b的年龄:"+b.getAge());
swap(a,b);
System.out.println("交换后:");
System.out.println("a的年龄:"+a.getAge()+"-----------b的年龄:"+b.getAge());
}
public void swap(Person x, Person y) {
Person temp = x;
x = y;
y = temp;
}
答案是互换不成功,依然是原来的值。
笔者开通了个人微信公众号【银河架构师】,分享工作、生活过程中的心得体会,填坑指南,技术感悟等内容,会比博客提前更新,欢迎订阅。