/*-- 例子 --*/

public class Test {

public static void main(String[] args) {
Integer a = 1;
Integer b = 2;

System.out.println("a 和 b 的原始的值:"+a+" "+b);

swap(a,b);

System.out.println("a 和 b 的现在的值:"+a+" "+b);
}

private static void swap(Integer a, Integer b) {
// TODO Auto-generated method stub

}

}

Java 形参和实参的区别:

形参​ :就是形式参数,用于定义方法的时候使用的参数,是用来接收调用者传递的参数的。 形参只有在方法被调用的时候,虚拟机才会分配内存单元,在方法调用结束之后便会释放所分配的内存单元。 因此,形参只在方法内部有效,所以针对引用对象的改动也无法影响到方法外。

实参​ :就是实际参数,用于调用时传递给方法的参数。实参在传递给别的方法之前是要被预先赋值的。 在本例中 swap 方法 的numa, numb 就是形参,传递给 swap 方法的 a,b 就是实参

注意:​在值传递调用过程中,只能把实参传递给形参,而不能把形参的值反向作用到实参上。在函数调用过程中,形参的值发生改变,而实参的值不会发生改变。而在引用传递调用的机制中,实际上是将实参引用的地址传递给了形参,所以任何发生在形参上的改变也会发生在实参变量上。

值传递和引用传递

JAVA的数据类型——分为两大类:基本类型 和 对象类型。相应的,变量也有两种类型:基本类型 和 引用类型。

基本类型的变量保存原始值,即它代表的值就是数值本身, 原始值一般对应在内存上的栈区;而引用类型的变量保存引用值,引用值指向内存空间的地址。代表了某个对象的引用,而不是对象本身。对象本身存放在这个引用值所表示的地址的位置。被引用的对象对应内存上的堆内存区。

基本类型包括: ​​byte​​, ​​short​​, ​​int​​, ​​long​​, ​​char​​, ​​float​​, ​​double​​, ​​boolean​​ 这八大基本数据类型; 引用类型包括: ​​类类型​​, ​​接口类型​​和 ​​数组。​

变量的基本类型和引用类型的区别:基本数据类型在声明时系统就给它分配空间

int a;

//虽然没有赋值,但声明的时候虚拟机就会 分配 4字节 的内存区域,

//而引用数据类型不同,它声明时只给变量分配了引用空间,而不分配数据空间:


String str;

//声明的时候没有分配数据空间,只有 4byte 的引用大小,

//在栈区,而在堆内存区域没有任何分配


str.length();

//这个操作就会报错,因为堆内存上还没有分配内存区域,而 a = 1; 这个操作就不会报错。

值传递:方法调用时,实际参数把它的值传递给对应的形式参数,函数接收的是原始值的一个copy, 此时内存中存在两个相等的基本类型,即实际参数和形式参数,后面方法中的操作都是对形参这个值的修改,不影响实际参数的值。

引用传递:也称为地址传递、址传递。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,函数接收的是原始值的内存地址在方法执行中,形参和实参内容相同,指向同一块内存地址,方法执行中对引用的操作将会影响到实际对象。

注意:​类似的 String, Integer, Float, Double, Short, Byte, Long, Character等基本包装类型类。因为他们本身没有提供方法去改变内部的值,例如 Integer内部有一个 value 来记录 int基本类型的值,但是没有提供修改它的方法,而且 也是 final类型的,无法通过 常规手段更改。所以虽然他们是引用类型的,但是可以认为它是值传递,这个也只是认为,事实上还是引用传递、址传递。

代码完善:

private static void swap(Integer numa, Integer numb){

int tmp = numa.intValue();

try{

Field field = Integer.class.getDeclaredField("value");

field.setAccessible(true);

field.set(numa, numb);

field.set(numb, new Integer(tmp));


}catch(Exception e){

e.printStackTrace();

}
}

 Integer的装箱操作,当给 Integer.value 赋值 int时,JVM 检测到 int不是Integer类型,需要装箱,才执行了Integer.valueOf()方法。而 field.set(numb,new Integer(tmp)) 设置的 是Integer类型了,就不会再拆箱后再装箱。