Java中的参数传递方式

最近在学习《JAVA核心技术》这本书,在看到第4章时对于方法参数的传递方式与C++中的传递方式发生了混淆。根本上讲java的传递方式是值传递,其在传递对象引用的时候类似于C++的指针传递方式;但并不是C++中的引用传递。这里引用了http://blog.sina.com.cn/s/blog_59ca2c2a0100qhjx.html 与http://hxraid.iteye.com/blog/428856 两篇文章来解释。

(一)

无论是什么语言,要讨论参数传递方式,就得从内存模型说起,主要是我个人觉得从内存模型来说参数传递更为直观一些。闲言少叙,下面我们就通过内存模型的方式来讨论一下Java中的参数传递。

这里的内存模型涉及到两种类型的内存:栈内存(stack)和堆内存(heap)。基本类型作为参数传递时,传递的是这个值的拷贝。无论你怎么改变这个拷贝,原值是不会改变的。看下边的一段代码,然后结合内存模型来说明问题:

public class ParameterTransfer {

 public static void main(String[] args) {
  int num = 30;
  System.out.println("调用add方法前num=" + num);
  add(num);
  System.out.println("调用add方法后num=" + num);
 }
 
 public static void add(int param) {
  param = 100;
 }
 
}

 

    这段代码运行的结果如下:

调用add方法前num=30
调用add方法后num=30

    程序运行的结果也说明这一点,无论你在add()方法中怎么改变参数param的值,原值num都不会改变。

    下边通过内存模型来分析一下。

    当执行了int num = 30;这句代码后,程序在栈内存中开辟了一块地址为AD8500的内存,里边放的值是30,内存模型如下图:

Java中的参数传递方式
    执行到add()方法时,程序在栈内存中又开辟了一块地址为AD8600的内存,将num的值30传递进来,此时这块内存里边放的值是30,执行param = 100;后,AD8600中的值变成了100。内存模型如下图:   
Java中的参数传递方式
    地址AD8600中用于存放param的值,和存放num的内存没有任何关系,无论你怎么改变param的值,实际改变的是地址为AD8600的内存中的值,而AD8500中的值并未改变,所以num的值也就没有改变。
    以上是基本类型参数的传递方式,下来我们讨论一下对象作为参数传递的方式。

    先看下边的示例代码:

public class ParameterTransfer {

 public static void main(String[] args) {
  String[] array = new String[] {"huixin"};
  System.out.println("调用reset方法前array中的第0个元素的值是:" + array[0]);
  reset(array);
  System.out.println("调用reset方法后array中的第0个元素的值是:" + array[0]);
 }
 
 public static void reset(String[] param) {
  param[0] = "hello, world!";
 }
 
}

运行的结果如下:

调用reset方法前array中的第0个元素的值是:huixin
调用reset方法后array中的第0个元素的值是:hello, world!

当对象作为参数传递时,传递的是对象的引用,也就是对象的地址。下边用内存模型图来说明。

Java中的参数传递方式

当程序执行了String[] array = new String[] {"huixin"}后,程序在栈内存中开辟了一块地址编号为AD9500内存空间,用于存放array[0]的引用地址,里边放的值是堆内存中的一个地址,示例中的值为BE2500,可以理解为有一个指针指向了堆内存中的编号为BE2500的地址。堆内存中编号为BE2500的这个地址中存放的才是array[0]的值:huixin。

当程序进入reset方法后,将array的值,也就是对象的引用BE2500传了进来。这时,程序在栈内存中又开辟了一块编号为AD9600的内存空间,里边放的值是传递过来的值,即AD9600。可以理解为栈内存中的编号为AD9600的内存中有一个指针,也指向了堆内存中编号为BE2500的内存地址,如图所示:

Java中的参数传递方式

这样一来,栈内存AD9500和AD9600(即array[0]和param的值)都指向了编号为BE2500的堆内存。

在reset方法中将param的值修改为hello, world!后,内存模型如下图所示:

Java中的参数传递方式

改变对象param的值实际上是改变param这个栈内存所指向的堆内存中的值。param这个对象在栈内存中的地址是AD9600,里边存放的值是BE2500,所以堆内存BE2500中的值就变成了hello,world!。程序放回main方法之后,堆内存BE2500中的值仍然为hello,world!,main方法中array[0]的值时,从栈内存中找到array[0]的值是BE2500,然后去堆内存中找编号为BE2500的内存,里边的值是hello,world!。所以main方法中打印出来的值就变成了hello,world!

无论是基本类型作为参数传递,还是对象作为参数传递,实际上传递的都是值,只是值的的形式不用而已。第一个示例中用基本类型作为参数传递时,将栈内存中的值30传递到了add方法中。第二个示例中用对象作为参数传递时,将栈内存中的值BE2500传递到了reset方法中。当用对象作为参数传递时,真正的值是放在堆内存中的,传递的是栈内存中的值,而栈内存中存放的是堆内存的地址,所以传递的就是堆内存的地址。这就是它们的区别。

补充一下,在Java中,String是一个引用类型,但是在作为参数传递的时候表现出来的却是基本类型的特性,即在方法中改变了String类型的变量的值后,不会影响方法外的String变量的值。关于这个问题,可以参考如下两个地址:

http://freej.blog.51cto.com/235241/168676

http://dryr.blog.163.com/blog/static/58211013200802393317600/

我觉得是这两篇文章中提到的两个原因导致的,一个是String实际上操作的是char[],可以理解为String是char[]的包装类。二是给String变量重新赋值后,实际上没有改变这个变量的值,而是重新new了一个String对象,改变了新对象的值,所以原来的String变量的值并没有改变。


(二)

 方法调用(call by) 是一个标准的计算机科学术语。方法调用根据参数传递的情况又分为值调用( call by reference ) 引用调用( call by value ) 。江湖上有很多关于这两种调用的定义 ,最通常的说法是传递值的是值调用,传递地址的是引用调用。这其实很不恰当,这种 这些说法很容易让我们联想到Java的对象参数传递是引用调用,实际上,Java的对象参数传递仍然是值调用 。 

      我们首先用一段代码来证实一下为什么Java的对象参数传递 是值调用。

Java代码   收藏代码
  1. public class Employee {  
  2.   
  3.     public String name=null;  
  4.       
  5.     public Employee(String n){  
  6.         this.name=n;  
  7.     }  
  8.     //将两个Employee对象交换  
  9.     public static void swap(Employee e1,Employee e2){  
  10.         Employee temp=e1;  
  11.         e1=e2;  
  12.         e2=temp;  
  13.                 System.out.println(e1.name+" "+e2.name); //打印结果:李四 张三  
  14.     }  
  15.     //主函数  
  16.     public static void main(String[] args) {  
  17.         Employee worker=new Employee("张三");  
  18.         Employee manager=new Employee("李四");  
  19.         swap(worker,manager);  
  20.         System.out.println(worker.name+" "+manager.name); //打印结果仍然是: 张三 李四  
  21.     }  
  22. }  

 

      上面的结果让人很失望,虽然形参对象e1,e2的内容交换了,但实参对象worker,manager并没有互换内容。这里面最重要的原因就在于形参e1,e2是实参worker,manager的地址拷贝。

      大家都知道,在Java中对象变量名实际上代表的是对象在堆中的地址(专业术语叫做对象引用 )。在Java方法调用的时候,参数传递的是对象的引用。重要的是,形参和实参所占的内存地址并不一样,形参中的内容只是实参中存储的对象引用的一份拷贝。

       如果大家对JVM内存管理中Java栈 局部变量区 有所了解的话(可以参见《 Java 虚拟机体系结构 》),就很好理解上面这句话。在JVM运行上面的程序时,运行main方法和swap方法,会在Java栈中先后push两个叫做栈帧的内存空间。main栈帧中有一块叫局部变量区的内存用来存储实参对象worker和manager的引用。而swap栈帧中的局部变量区则存储了形参对象e1和e2的引用。虽然e1和e2的引用值分别与worker和manager相同,但是它们占用了不同的内存空间。当e1和e2的引用发生交换时,下面的图很清晰的看出完全不会影响worker和manager的引用值。

             

      Java对象参数传递虽然传递的是地址(引用),但仍然是值调用。是时候需要给引用调用和值调用一个准确的定义了。

 

      值调用(call by value)  在参数传递过程中,形参和实参占用了两个完全不同的内存空间。形参所存储的内容是实参存储内容的一份拷贝。实际上,Java对象的传递就符合这个定义,只不过形参和实参所储存的内容并不是常规意义上的变量值,而是变量的地址。咳,回过头想想:变量的地址不也是一种值吗!

      引用调用(call by reference)  在参数传递的过程中,形参和实参完全是同一块内存空间,两者不分彼此。实际上,形参名和实参名只是编程中的不同符号,在程序运行过程中,内存中存储的空间才是最重要的。不同的变量名并不能说明占用的内存存储空间不同。

 

      大体上说,两种调用的根本并不在于传递的是值还是地址(毕竟地址也是一个值),而是在于形参和实参是否占用同一块内存空间。事实上,C/C++的指针参数传递也是值调用,不信试试下面的C代码吧!

C代码   收藏代码
  1. #include<stdio.h>  
  2. void swap(int *a1,int *b1){  
  3.     int *t=a1;  
  4.     a1=b1;  
  5.     b1=t;  
  6. }  
  7. int main(){  
  8.     int x1=100;  
  9.     int x2=200;  
  10.         int *a=&x1;  
  11.     int *b=&x2;  
  12.     printf("%d %d\n",*a,*b);  
  13.     swap(a,b);  
  14.     printf("%d %d\n",*a,*b);  
  15.     return 0;  
  16. }  

         但C/C++是有引用调用的,这就是C/C++一种叫做引用的变量声明方法: int a; int &ra=a; 其中ra是a的别名,两者在内存中没有区别,占用了同一个内存空间。而通过引用(别名)的参数传递就符合引用调用的特点了。大家可以去试试

void swap(int &a1,int &b1);的运行结果。



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值