网络上有太多关于JAVA参数传递是传值还是传引用的讨论,其实大多是概念不清,混淆视听。从程序运行的角度来看,参数传递,只有传值,从不传递其它的东西。只不过值的内容有可能是数据,也有可能是一个内存地址。

 Java中的数据类型有两大类,即基本类型(primitive types), 共有8种,包括int, short, long, byte, float, double, boolean, char,存在于栈(Stack)中。另一种暂称为对象类型,包括Integer, String, Double等相应基本数据类型的包装类,以及其他所有JAVA自带和用户自定义的类,这些类数据全部存在于堆中(Heap),如下图所示。

  对象类型的数据不同于基本类型的数据,我们所定义的对象变量并不是对象实例本身,而是对象的一个引用(reference)地址,该内存地址用来定位该对象实例在HEAP中的位置。对象实例本身和对象的引用分别保存在HEAP和STACK中,对象引用和对象实例之间的关系好比遥控器和电视机之间的关系,在房间走动的时候里,你只需拿着遥控器就可以控制电视机,而不必带着电视机。而且,即使没有电视机,遥控器也可以独立存在,也就是说你可以定义一个对象类型的变量,但是可以暂时不和一个对象实例关联。多个对象引用也可以指向同一个对象实例。


   

    

    

JVM中的功能

          内存数据区    

             

存储数据

对象实例

基本数据类型, 指令代码,常量,对象的引用地址


下面我们来看看几个例子,您就会更加明白。

例子1:

public class Test {
 public static void changeValue(int i) {
 i=2;
 System.out.println("during test: i = " + i);
 }
 public static void main(String[] args) {
 int i = 1;
 System.out.println("Before test: i = " + i);
 changeValue(i);
 System.out.println("After test: i = " + i);
 }
 }


  运行结果:
Before test: i = 1
during test: i = 2
After test: i = 1

不难看出,虽然在 changeValue(int i)方法中改变了传进来的参数的值,但对这个参数源变量本身并没有影响。其内部原理是,main方法里的变量和changeValue方法的参数是两个不同的变量,以参数形式传递简单类型的变量时,实际上是将参数的值作了一个拷贝传进方法的,那么在方法里再怎么改变其值,其结果都是只改变了拷贝的值,而不是源值。

例子2:

public class Test {
 public static void test(StringBuffer str) {
 str.append(", World!");
 }
 public static void main(String[] args) {
 StringBuffer string = new StringBuffer("Hello");
 test(string);
 System.out.println(string);
 }
 }


  运行结果:
Hello, World!

 

例子3:

public class Test {
 public static void test(String str) {
 str = "World";
 }
 public static void main(String[] args) {
 String string = "Hello";
 test(string);
 System.out.println(string);
 }
 }

运行结果:

Hello。

这个结果和上面结果矛盾吗?一点也不矛盾。在这个例子中,参数传递过程和上个例子完全一样,但是在test方法里并不是对原来指向的对象实例进行操作,而是把str指向了另外一个对象实例,当然对原来的对象实例毫无影响。