咱们首先用一段代码来证明一下为何Java的对象参数传递 是值调用。 html
public class Employee {
public String name=null;
public Employee(String n){
this.name=n;
}
//将两个Employee对象交换
public static void swap(Employee e1,Employee e2){
Employee temp=e1;
e1=e2;
e2=temp;
System.out.println(e1.name+" "+e2.name); //打印结果:李四 张三
}
//主函数
public static void main(String[] args) {
Employee worker=new Employee("张三");
Employee manager=new Employee("李四");
swap(worker,manager);
System.out.println(worker.name+" "+manager.name); //打印结果仍然是: 张三 李四
}
}
上面的结果让人很失望,虽然形参对象e1,e2的内容交换了,但实参对象worker,manager并无互换内容。这里面最重要的缘由就在于形参e1,e2是实参worker,manager的地址拷贝。
你们都知道,在Java中对象变量名实际上表明的是对象在堆中的地址(专业术语叫作对象引用 )。在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代码吧! java
#include<stdio.h>
void swap(int *a1,int *b1){ EMS指南
int *t=a1;
a1=b1;
b1=t;
}
int main(){
int x1=100;
int x2=200;
int *a=&x1;
int *b=&x2;
printf("%d %d\n",*a,*b);
swap(a,b);
printf("%d %d\n",*a,*b);
return 0;
}
编程