• Java 中的传值与传引用


    Java 函数中的传值和传引用问题一直是个比较“邪门”的问题,其实 Java 函数中的参数都是传递值的,所不同的是对于基本数据类型传递的是参数的一份拷贝,对于类类型传递的是该类参数的引用的拷贝,当在函数体中修改参数值时,无论是基本类型的参数还是引用类型的参数,修改的只是该参数的拷贝,不影响函数实参的值,如果修改的是引用类型的成员值,则该实参引用的成员值是可以改变的,例子如下

    class Model {
        public int i = 0;
        public String s = "no value";
    
        public static void changeInt(int i) {// 改变int型变量的函数
            i = 100;
        }
    
        public static void changeString(String s) {// 改变String型变量的函数
            s = "changeString";
        }
    
        public static void changeModel(Model model) {// 改变Model型变量的函数
            model = new Model();
            model.i = 1;
            model.s = "changeModel";
        }
    
        public static void changeModel2(Model model) {// 改变Model型变量的成员的函数
            model.i = 1;
            model.s = "changeModel";
        }
    
        // 测试程序
        public static void main(String[] args) {
            int i = 0;
            String s = "hello";
            Model model_1 = new Model();
            Model model_2 = new Model();
    
            changeInt(i);
            System.out.println("i = " + i);
            changeString(s);
            System.out.println("s = " + s);
            changeModel(model_1);
            System.out.println("model_1.s = " + model_1.s);
            changeModel2(model_2);
            System.out.println("model_2.s = " + model_2.s);
        }
    }
    

    测试结果

    "C:Program FilesJavajdk1.8.0_201injava.exe"
    i = 0 // 未改变
    s = hello // 未改变
    model_1.s = no value // 未改变
    model_2.s = changeModel // 改变
    
    Process finished with exit code 0
    

    可以看出 i 没有改变,s 也没有改变,model_1 也没有改变,model_2的 s 改变了。

    总结

    • Java 中的形参是复制实参的一份拷贝(在栈中的拷贝,对于引用型则是复制引用的拷贝)
    • 所以在函数中改变形参是无法改变实参的值的,改变引用只是将形参所代表的引用指向另外的新的对象,而实参的引用还指向原来的对象
    • 改变形参引用的成员当然会影响实参引用成员的值,因为他们的引用都指向同一个对象。

    这里有个陷阱,我最开始没转过来

    按照前面的例子,String应该是一个封装类型,它应该是引用传递,是可以改变值得, 运行的结果应该是changeString才对。String也不是基本类型,这就矛盾了。

    这就要从Java底层的机制讲起了,Java的内存模型分为"堆"和"栈" 。

    • 基本类型,变量放在栈里

      1. 虚拟机分配给num一个内存地址,并且存了一个值0
      2. 虚拟机复制了一个num给函数,我们叫他num2,num2和num的内存地址不同,但存的值都是0
      3. 虚拟机将num2传入方法,方法将num2的值改为1
      4. 方法结束,方法外打印num的值,由于num在内存中的值没有改变,还是0,所以打印是0
    • 封装类型,对象放在堆里,对象的引用放在栈里。

      1. 虚拟机在堆中开辟了一个Product的内存空间,内存中包含proName和num
      2. 虚拟机在栈中分配给对象p一个内存地址,这个地址中存的是步骤1堆中Product的地址
      3. 虚拟机复制了一个p给函数,我们叫他p2。p和p2的内存地址不同,但它们存的值是相同的,都是1中Product的内存地址
      4. 将p2传入方法,方法改变了步骤1中的proName和num
      5. 方法结束,方法外打印p中变量的值,由于p和p2中存的都是1中Product的地址,但是步骤1中Product里的值发生了改变, 所以,方法外打印p的值,是方法执行以后的。我们看到的效果是封装类型的值是改变的
    • 但String又有些特别

      1. 虚拟机在堆中开辟一块内存,并存值"ab"
      2. 虚拟机在栈中分配给str一个内存,内存中存的是1中的地址
      3. 虚拟机复制一份str,我们叫str2,str和str2内存不同,但存的值都是步骤1的地址
      4. 将str2传入方法体
      5. 方法体在堆中开辟一块内存,并存值"cd"
      6. 方法体将str2的值改变,存入5的内存地址
      7. 方法结束,方法外打印str,由于str存的是步骤1的地址,所有打印结果是"ab"

    所以,String并不是一个基本类型,也不是有什么特殊处理,s = "cd"可以理解为s = new String("cd"),所以String的实参自然是不变。

    结论:Java 中的形参是复制实参的一份拷贝

    • 基本类型,形参和实参是值相同2个变量
      • 形参的改变不会影响实参
    • 封装类型,形参是和实参引用相同2个变量
      • 形参指向新的引用之前,改变形参的field是会影响到实参
      • 形参指向新的引用之后(a = new Apple()a = "cd"等),实参不会受到影响

    参考

    https://www.cnblogs.com/zhangj95/p/4184180.html

    https://www.cnblogs.com/boboooo/p/9066831.html

  • 相关阅读:
    Nginx学习总结(一)
    zabbix3.4.8配置自动发现主机并监控
    Windows server 2012/2016系统安装zabbix3.2客户端
    CentOS7.6系统安装zabbix3.4.8客户端
    一个小爬虫的整体解决方案
    如何通过一个立方体搭建一栋楼
    用Scrapy框架开发的一个爬虫项目
    寻找替代imagemin更好的插件
    原生和es6复杂数组去重的方法
    javascript关于对象或者数组深克隆的写法
  • 原文地址:https://www.cnblogs.com/n031/p/11660767.html
Copyright © 2020-2023  润新知