• 【解惑】Java方法参数是引用调用还是值调用?


     最近找实习,在笔试时候做了下面这道题:求下面代码的输出结果。

    public class MyClass {

    static void aMethod(StringBuffer sf1,StringBuffer sf2){
    sf1.append(sf2);
    sf2=sf1;
    }

    public static void main(String[] args){
    StringBuffer sf1=new StringBuffer("A");
    StringBuffer sf2=new StringBuffer("B");
    aMethod(sf1,sf2);
    System.out.println(sf1+":"+sf2);
    }
    }

    我当时想到的输出结果是:   AB:AB。   很显然掉进出题人的圈套里边了,正确答案应该是AB:B。

    由于长时间没看JAVA,可能有些知识点生疏了,这道题就考了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 ",*a,*b);  
    13.     swap(a,b);  
    14.     printf("%d %d ",*a,*b);  
    15.     return 0;  
    16. }  

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

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

  • 相关阅读:
    如何把textfield或者textview中长按出现的(全选,复制,粘贴)显示成中文
    免费真机调试 -- Xcode7
    Android性能测试工具 Emmagee
    iOS 开发 入门:使用Ad Hoc 进行用户测试
    栈与队列的区别
    iOS中ASI和AFN的区别
    iOS开发之监测网络状态
    xcode设置项目图标玻璃镜效果
    isEqual,isEqualTostring,==三者的区别
    iphone匹配邮箱的正则表达式
  • 原文地址:https://www.cnblogs.com/ambitious-kevin/p/4512141.html
Copyright © 2020-2023  润新知