Java学习者论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

手机号码,快捷登录

恭喜Java学习者论坛(https://www.javaxxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,购买链接:点击进入购买VIP会员
JAVA高级面试进阶视频教程Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程

Go语言视频零基础入门到精通

Java架构师3期(课件+源码)

Java开发全终端实战租房项目视频教程

SpringBoot2.X入门到高级使用教程

大数据培训第六期全套视频教程

深度学习(CNN RNN GAN)算法原理

Java亿级流量电商系统视频教程

互联网架构师视频教程

年薪50万Spark2.0从入门到精通

年薪50万!人工智能学习路线教程

年薪50万!大数据从入门到精通学习路线年薪50万!机器学习入门到精通视频教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程 MySQL入门到精通教程
查看: 259|回复: 0

[默认分类] Java函数参数传递方式详解

[复制链接]
  • TA的每日心情
    开心
    2021-12-13 21:45
  • 签到天数: 15 天

    [LV.4]偶尔看看III

    发表于 2018-6-21 10:08:57 | 显示全部楼层 |阅读模式


    转:http://zzproc.iteye.com/blog/1328591
    在阅读本文之前,根据自己的经验和理解,大家可以先思考并选择一下java函数的参数传递方式:
    A. 是按值传递的?
    B. 按引用传递的?
    C. 部分按值部分按引用?
    此处暂不宣布正确答案,我们通过一个简单的例子让大家自己找答案:
    1. 先定义一个类型Value

      
       
       
         Java代码  
       
       
       
      
        public static class Value {  
            private String value = "value";  
            public String getValue() { return value; }  
            public void setValue(String value) { this.value = value; }  
        }  
      
      


    2. 写两个函数newValue和modifyValue:newValue会将入参指向一个新的对象,modifyValue会调用入参的setValue方法修改对象的value值。


      
       
       
         Java代码  
       
       
       
      
        public static void newValue(Value value) {  
            value = new Value();  
            value.setValue("new value");  
            System.out.println("In newValue, HashCode = " + value.hashCode() + ", value = " + value.getValue());  
        }  
             
        public static void modifyValue(Value value) {  
            value.setValue("new value");  
            System.out.println("In modifyValue, HashCode = " + value.hashCode() + ", value = " + value.getValue());  
        }  
      
      


    3. 简单的测试代码


      
       
       
         Java代码  
       
       
       
      
        public static void main(String[] args) {  
            Value value1 = new Value();  
            System.out.println("Before modify, HashCode = " + value1.hashCode() + ", value = " + value1.getValue());  
            // 将value1指向新的Value对象  
            newValue(value1);  
            System.out.println("After modify, HashCode = " + value1.hashCode() + ", value = " + value1.getValue() + "\n");  
            Value value2 = new Value();  
            System.out.println("Before modify, HashCode = " + value2.hashCode() + ", value = " + value2.getValue());  
            // 使用object的set方法,修改对象的内部值  
            modifyValue(value2);  
            System.out.println("After modify, HashCode = " + value2.hashCode() + ", value = " + value2.getValue());  
        }  
      
      


    4. 执行结果日志:


      
       
       
         Java代码  
       
       
       
      
        Before modify, HashCode = 12677476, value = value  
        In newValue, HashCode = 33263331, value = new value  
        After modify, HashCode = 12677476, value = value  
          
        Before modify, HashCode = 6413875, value = value  
        In modifyValue, HashCode = 6413875, value = new value  
        After modify, HashCode = 6413875, value = new value  
      
      




    5. 结果分析:


    上述代码这是非常常见的一种编程模式:在外围定义|保存|获取一个值或对象,将这个对象作为参数传入一个方法,在方法中修改对象的属性、行为。但两个方法newValue和modifyValue的修改方式不一样,在方法调用之后,该对象在外围看来也有很大的差别!如何理解这种差异呢?先温故一下按值传递、按引用传递的概念:


    * 按值传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的一个副本。因此,如果函数修改了该参数,仅改变副本,而原始值保持不变。


    * 按引用传递意味着当将一个参数传递给一个函数时,函数接收的是原始值的内存地址,而不是值的副本。因此,如果函数修改了该参数,参数的原始值(函数块之外的调用代码中)也随之改变。


    正确的答案:A——Java函数是按值传递参数的!
      


    分析一下日志:


    * 第一段日志输出,value1参数在newValue方法内部被改为指向新对象,并输出了新对象的hashCode和value值,但跳出newValue方法域之后,在main方法中的value1没有发生任何变化,这符合按值传递的定义和特点;如果是按引用传递,value1在调用newValue(Value value)方法之后,应该是发生变化的。


    * 第二段日志输出,value2在modifyValue方法内部进行了setValue操作,hashCode不变而value被修改,离开modifyValue方法域之后,在main方法中value2确实发生了变更。使用过C++的人容易将这种现象理解为:按引用传递函数参数!因为这跟C++中的按引用传递像极了!但这里恰恰是最容易陷入误区的地方!


    两段日志的不同现象背后所隐藏的是原理是:Java语言是按值传递参数,按引用传递对象的;Java中所操作的对象其实都是操作对象的引用,object本身保存在“堆”中,而对象的“引用“保存在寄存器或“栈”中。


    伪代码描述一下newValue方法和modifyValue方法的不同之处:


      
       
       
         Java代码  
       
       
       
      
        newValue{  
            Value_ref2 = value_ref1;    // 按值传入引用value_ref1,得到value_ref1的副本  
            value_obj2 = new Value();   // value_obj2被创建、初始化在“堆“中  
            value_ref2 -> value_obj2;    // value_ref2 指向value_obj2  
        value_ref2 ->value_obj2.setValue(“xxx”); // value_obj2 的value被修改  
        printValueObj2();           // 此处打印的是obj2的值  
        }  
        modifyValue{  
            Value_ref2 = value_ref1;    // 按值传入引用value_ref1,得到value_ref1的副本  
        value_ref2 ->value_obj1.setValue(“xxx”); // value_obj1 的value被修改  
        printValueObj1();           // 此处打印的是obj1的值  
        }  
      
      


    够清楚了吧!value1_ref1在作为参数传入函数的时候,首先被复制了一份副本value1_ref2供函数域使用,此时这两个ref都是指向同一个value_obj; newObject函数中的代码[ value = new Value(); ] 其实是将value1_ref1指向了一个新的对象value_obj2;在这之后的set操作都是对新对象的操作;modifyValue函数是通过set方法直接操作value_obj1,这是跟newValue函数的不同之处。




    如果还是不太明白,请先确定是否已经理解 “引用”、“对象”的概念,可以Google、百度相应的文章学习一下:)


      
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|Java学习者论坛 ( 声明:本站资料整理自互联网,用于Java学习者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-3-28 19:12 , Processed in 0.586219 second(s), 46 queries .

    Powered by Discuz! X3.4

    © 2001-2017 Comsenz Inc.

    快速回复 返回顶部 返回列表