/*-- 例子 --*/
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类型了,就不会再拆箱后再装箱。