`
beston
  • 浏览: 8107 次
  • 性别: Icon_minigender_2
  • 来自: 上海
社区版块
存档分类
最新评论

[转]Java方法参数是引用调用还是值调用?

    博客分类:
  • Java
阅读更多

原文:http://hxraid.iteye.com/blog/428856

      方法调用(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);的运行结果。

 

-----------------------------------------------------------------------------------------------------------------------

public class StackTest {
	public static void main(String[] args){
		List<String> list = new ArrayList<String>();
		 list.add("111");
		 list.add("222");
		 list.add("333");
		 update(list);
		 System.out.println(list);
	}
	
	public static void update(List<String> list){
		List<String> result = new ArrayList<String>();
		 result.add("111");
		 result.add("222");
		 list.remove(0);
		 list = result;
	}
}

 

这里补充一个知识:一个java程序运行过程中,所有方法开辟的对象的内容都保存在唯一的一段内存中,这个内存我们叫“堆”,而每个方法运行时的局部变量和对象的引用全部短暂的保存在JVM为方法创建的独立的栈桢中,每个方法对应一个栈桢,所有栈桢存放在一段内存空间中,我们叫“堆栈”。

我们分析一下过称:
(1)main函数首先在内存堆中开辟了一个ArrayList的对象list,里面存放的是[111,222,333],这个对象list在堆中有一个地址,不妨假设是0x1111。
(2)然后要注意了:main函数调用update函数传递的是对象list的引用,也就是0x1111这样一个值,这个值保存在内存栈空间的update栈桢中。,这个值传给了update的参数list,也就是说在update这个方法的栈桢中也开辟了一个空间,存放的是main传递过来的0x1111这个值。
(3) 在update运行时也会在内存堆中开辟一个ArrayList的对象result,里面存放的是[111,222],这个对象的地址我们假设为Ox2222。这个值也是存放在update的栈桢中。
此时要注意了,update栈桢中有两个值,一个是形参list的0x1111,另一个是刚开辟的result的Ox2222。
(4) 当执行list.remove("111")的时候,JVM是将栈桢中0x1111地址所指向的内存堆中的对象[111,222,333]中的一个元素"111"删除。这句执行之后,Ox1111地址中的内容已经少了一个"111"了。
(5) 比较迷惑的是下面的赋值语句:list = result;
    这句话是引用赋值,只是将update栈桢中原本存放Ox1111这个值的位置赋值成了Ox2222。而并没有将Ox2222所指向的内存堆中的[111,222]对象全部覆盖掉Ox1111所指向的内存堆中的[222,333],注意Ox1111所指向的对象内容已经被第(4)步remove掉了一个。
    更重要的是,main主函数栈桢中的存放Ox1111这个值的位置并没有被赋值掉,所以在main中最后打印的仍然是Ox1111所指向的对象[222,333]

这次方法调用仍然是引用传递,是值传递,而非地址传递。想搞清楚JVM内部的运行情况,建议看《深入JVM》。

分享到:
评论

相关推荐

    Java方法参数是引用调用还是值调用?

    Java方法参数传递机制是值调用,而不是引用调用,这一点常常引起误解。在Java中,对象作为参数传递时,传递的是对象引用的副本,而不是对象本身或其内存地址的副本。这意味着,即使在方法内部改变了对象引用,原始...

    java 之方法调用 方法传参 值传递还是引用传递字节码

    在Java中,方法是一组完成特定任务的代码块,可以通过方法名来调用执行。方法调用的基本语法是`methodName(参数列表)`。例如: ```java public void printMessage(String msg) { System.out.println(msg); } // ...

    XSLT 调用 Java 的类方法

    2. **调用Java方法**:在XSLT模板中通过指定的前缀和完整的类名加方法名来调用Java方法,并传入相应的参数。 ```xml &lt;xsl:value-of select='java:XsltFunctions.replace(.)'/&gt; ``` #### 四、实例演示 为了更好...

    java用JNA调用dll实例,包含各种参数调用

    你需要继承Struct类,定义成员变量并使用`Structure.ByValue`或`Structure.ByReference`来选择是值传递还是引用传递。 4. **结构体的指针和引用**:如果DLL函数需要结构体的指针,你可以创建一个指向Struct实例的...

    C#调用JAVA方法

    4. **调用Java方法**:使用`JNIEnv.CallObjectMethod`、`JNIEnv.CallVoidMethod`等方法,根据返回值类型和参数类型选择合适的方法。 5. **处理结果**:将Java对象的结果转换回.NET对象,以便在C#中进一步处理。 **...

    C#调用JavaWebService

    然后自动产生代理类,但是在调用JAVA的WebService时并没有这么简单,特别是对于SoapHeader的处理,通过C#添加Web引用方式访问JavaWebService的方法,除了string类型能正常传递参数外,q其他类型的参数不是默认值就是...

    C#调用java类、jar包方法

    2. 调用Java对象的方法,确保正确处理参数类型和返回值类型。例如,如果`MyClass`有一个名为`myMethod`的方法,接受一个字符串参数并返回一个整数: ```csharp var method = myClass.GetDeclaredMethod("myMethod", ...

    java方法的参数传递其二.docx

    Java 方法参数传递之引用类型 Java 方法参数传递是 Java 编程语言中的一种基本机制,在方法调用时将参数传递给方法体内的变量,但是在传递过程中,参数的类型会对传递结果产生影响。在本文中,我们将讨论 Java 方法...

    Java:按值传递还是按引用传递详细解说

    1. **参数是原始值的副本**:无论是基本数据类型还是对象的引用,传递的都是它们的副本。 2. **互不相关**:一旦传递完成,传递给函数或方法的参数与原始值不再有关联,即对参数所做的任何修改不会反映在原始值上。 ...

    Java语言的方法参数浅析

    通过对典型程序的研究与分析可以看出JAVA语言的参数传递总是传值调用的,但是对于基本类型的参数和对象类型的参数来说,参数传递的情况不完全相同.Java语言不能直接使用传引用调用,但是可以通过数组的方式模拟传引用...

    java调用window操作系统文本转语音并生成播放文件

    在Java编程环境中,调用Windows操作系统来实现文本转语音(Text-to-Speech, TTS)并生成播放文件是一项常见的需求。这项技术可以帮助开发者为应用程序添加语音合成功能,尤其适用于无障碍应用、语音助手或者多媒体...

    java调用Kettle引用jar包.zip

    Java调用Kettle是将Java程序与Pentaho Kettle(也称为Spoon)集成,以便利用Kettle的强大ETL(提取、转换、加载)能力。Kettle是一款开源的数据集成工具,它允许开发者通过编写XML脚本来执行数据处理任务。在Java...

    java rmi远程方法调用 客户端

    Java RMI(Remote Method Invocation,远程方法调用)是Java平台提供的一种分布式计算技术,它允许在不同的Java虚拟机之间透明地调用对象的方法。在RMI架构中,客户端能够像调用本地对象一样调用远程服务器上的对象...

    Java实验-掌握方法的声明和调用 掌握方法的值传递 掌握方法的重载

    本实验报告涵盖了 Java 编程语言中方法的声明、调用、值传递、重载和 Math 类的使用。通过三个实验,学生掌握了方法的声明和调用、值传递和方法的重载,并熟悉了 Math 类的使用。 一、方法的声明和调用 在 Java 中...

    DELPHI 11调用JAVA 接口

    调用Java方法时,传递这些参数。Java方法执行完成后,结果会返回,同样需要从JNI类型转换回Delphi类型。 6. **异常处理** Java的异常在JNI层会转换为本地异常,因此在Delphi中需要捕获并处理这些异常。 7. **资源...

    JNI与Java方法的相互调用

    `JNIEnv *env`参数提供了访问Java对象和调用Java方法的接口,`jobject this`是Java对象的引用。 要调用Java方法,我们可以使用`env`指针提供的函数,比如`CallVoidMethod()`,它允许我们在本地代码中调用Java方法。...

    Java-远程方法调用RMI参数详解.docx

    Java 远程方法调用(Remote Method Invocation,RMI)是一种在分布式环境中调用对象方法的技术,它使得Java应用程序能够透明地调用运行在不同 JVM 上的远程对象的方法。RMI 包含了一系列的参数,这些参数对于优化...

    Android之c调用java方法

    这里的`JNIEnv* env`参数用于访问JNI接口,`jobject obj`是Java对象的引用。 例如,如果你要在C代码中调用Java类的`sayHello()`方法,可以声明如下: ```c JNIEXPORT void JNICALL Java_...

    亲测可用,java 成功调用dll函数。包含调用回调函数,springboot版本。最近由于公司业务需要,要调用dll文件,用JNA调用。

    在Java通过JNA调用DLL时,如果DLL函数需要一个回调函数作为参数,那么Java需要定义一个接口,该接口的方法将作为回调函数的实现,JNA会处理这个接口的调用转换。 4. **Spring Boot**:Spring Boot是基于Spring框架...

Global site tag (gtag.js) - Google Analytics