• java の 请注意:java中没有引用传递


    方法调用(call by)是一个标准的计算机科学术语。方法调用根据参数传递的情况又分为值调用(call by reference)引用调用(call by value)

    先来我们来看如下代码: 

    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方法调用的时候,参数传递的是对象的引用。重要的是,形参和实参所占的内存地址并不一样,形参中的内容只是实参中存储的对象引用的一份拷贝。

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

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

    而是变量的地址。咳,回过头想想:变量的地址不也是一种值吗!

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

     注:转自http://hxraid.iteye.com/blog/428856

  • 相关阅读:
    C与设计模式---观察者模式
    如何在嵌入式产品中应用键值存储数据库
    Matlab 常用函数小结
    【Qt点滴】游戏2048
    经典ICP算法的问题
    基于矩阵分解的推荐系统实例
    【Qt点滴】UDP协议实例:简易广播实现
    【Qt点滴】:获取本机网络信息
    51单片机:光立方
    STM32单片机:四旋翼飞行器的飞控实现
  • 原文地址:https://www.cnblogs.com/zyoohoo/p/2545466.html
Copyright © 2020-2023  润新知