转载声明:本文转载自公众号「码匠笔记」。
前几天在头条上看到一道经典面试题,引发了一些思考。也是写这篇文章的导火索。
背景
请看题:
publicclassMain{
publicstaticvoidmain(String[]args){
Integera=1;
Integerb=2;
System.out.println("a="+a+",b="+b);
swap(a,b);
System.out.println("a="+a+",b="+b);
}
privatestaticvoidswap(Integernuma,Integernumb){
//请实现
}
}
看到这个题后 瞬间觉得有坑。也觉得为什么要书写一个 swap方法呢?如下实现不是更简单:
publicstaticvoidmain(String[]args){
Integera=1;
Integerb=2;
System.out.println("a="+a+",b="+b);
Integertmp=a;
a=b;
b=tmp;
System.out.println("a="+a+",b="+b);
}
输出:
a=1,b=2
a=2,b=1
完美实现交换。但是请注意,这是一道面试题,要的就是考验一些知识点。所以还是老老实实的实现 swap方法吧。 有的同学可能会想, Integer 是一个包装类型,是对Int的装箱和拆箱操作。其实也是一个对象。既然是对象,直接更改对象的引用不就行了?思路没问题,我们首先看看实现:
privatestaticvoidswap(Integernuma,Integernumb){
Integertmp=numa;
numa=numb;
numb=tmp;
System.out.println("numa="+numa+",numb="+numb);
}
输出:
a=1,b=2
numa=2,numb=1
a=1,b=2
不出意外,没有成功这是什么原因呢?
技术老手一看就知道问题出在形参和实参混淆了
JAVA的形参和实参的区别:
形参 顾名思义:就是形式参数,用于定义方法的时候使用的参数,是用来接收调用者传递的参数的。
形参只有在方法被调用的时候,虚拟机才会分配内存单元,在方法调用结束之后便会释放所分配的内存单元。
因此,形参只在方法内部有效,所以针对引用对象的改动也无法影响到方法外。
实参 顾名思义:就是实际参数,用于调用时传递给方法的参数。实参在传递给别的方法之前是要被预先赋值的。
在本例中 swap 方法 的numa, numb 就是形参,传递给 swap 方法的 a,b 就是实参
注意:在 值传递调用过程中,只能把实参传递给形参,而不能把形参的值反向作用到实参上。在函数调用过程中,形参的值发生改变,而实参的值不会发生改变。而在 引用传递调用的机制中,实际上是将实参引用的地址传递给了形参,所以任何发生在形参上的改变也会发生在实参变量上。那么问题来了,什么是 值传递和 引用传递
值传递和引用传递
在谈 值传递和 引用传递之前先了解下 Java的数据类型有哪些
JAVA的数据类型
Java 中的数据类型分为两大类, 基本类型和 对象类型。相应的,变量也有两种类型: 基本类型和 引用类型 基本类型的变量保存 原始值,即它代表的值就是数值本身, 原始值一般对应在内存上的 栈区而 引用类型的变量保存 引用值, 引用值指向内存