Java函数参数传递方式详解
在阅读本文之前,根据自己的经验和理解,大家可以先思考并选择一下Java函数的参数传递方式: 
A. 是按值传递的? 
B. 按引用传递的? 
C. 部分按值部分按引用? 
此处暂不宣布正确答案,我们通过一个简单的例子让大家自己找答案: 
1. 先定义一个类型Value
- 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值。
- 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. 简单的测试代码
- 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. 执行结果日志:
- 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方法的不同之处:
- 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、百度相应的文章学习一下:)
Java函数参数传递方式详解的更多相关文章
- Java 函数参数传递方式详解                                                    分类:            Java Game             2014-08-15 06:34    82人阅读    评论(0)    收藏
		
转:http://zzproc.iteye.com/blog/1328591 在阅读本文之前,根据自己的经验和理解,大家可以先思考并选择一下Java函数的参数传递方式: A. 是按值传递的? B. ...
 - C++中函数调用时的三种参数传递方式详解
		
在C++中,参数传递的方式是“实虚结合”. 按值传递(pass by value) 地址传递(pass by pointer) 引用传递(pass by reference) 按值传递的过程为:首先计 ...
 - 基于.Net Framework 4.0 Web API开发(2):ASP.NET Web APIs 参数传递方式详解
		
概述: ASP.NET Web API 的好用使用过的都知道,没有复杂的配置文件,一个简单的ApiController加上需要的Action就能工作.调用API过程中参数的传递是必须的,本节就来谈谈 ...
 - js-arguments 函数参数对象详解
		
前言 JavaScript 函数参数不同于其他编程语言,既不需要规定参数的类型,也不需要关心参数的个数,因此 JavaScript 因函数参数而变得十分灵活,本文总结一下 arguments 参数对象 ...
 - Python中scatter函数参数用法详解
		
1.scatter函数原型 2.其中散点的形状参数marker如下: 3.其中颜色参数c如下: 4.基本的使用方法如下: #导入必要的模块 import numpy as np import matp ...
 - java newInstance() 的参数版本与无参数版本详解
		
newInstance() 的参数版本与无参数版本详解 博客分类: Core Java 通过反射创建新的类示例,有两种方式: Class.newInstance() Constructor.new ...
 - java反射机制深入详解
		
java反射机制深入详解 转自:http://www.cnblogs.com/hxsyl/archive/2013/03/23/2977593.html 一.概念 反射就是把Java的各种成分映射成 ...
 - Scala 深入浅出实战经典 第60讲:Scala中隐式参数实战详解以及在Spark中的应用源码解析
		
王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-87讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...
 - Java 8 Lambda 表达式详解
		
一.Java 8 Lambda 表达式了解 参考:Java 8 Lambda 表达式 | 菜鸟教程 1.1 介绍: Lambda 表达式,也可称为闭包,是推动 Java 8 发布的最重要新特性. La ...
 
随机推荐
- Mysql下在某一列后即表的某一位置添加新列的sql语句
			
Mysql简介 MySQL是一个开放源码的小型关联式数据库管理系统,开发者为瑞典MySQL AB公司.MySQL被广泛地应用在Internet上的中小型网站中.由于其体积小.速度快.总体拥有成本低,尤 ...
 - hpple 简单使用
			
最近项目使用到hpple,简单说一下使用方式,做做笔记 let responseData = response as! NSData let utf8Html = responseData.strin ...
 - 尚学堂马士兵Oracle教程笔记
			
检查Oracle安装 首先,以超级管理员的身份登录oracle sqlplus sys/bjsxt as sysdba 然后,解除对scott用户的锁 alter user scott account ...
 - e+开发中的各种问题
			
1.数据交换后走的查询公式还是controller所配置的公式
 - Docker之配置Centos_ssh
			
写Dockerfile配置文件 #DockerfileFROM centos:6 #以下命令用在什么镜像中MAINTAINER cuizhipeng <cuizhipeng@126.com&g ...
 - Ubuntu 之旅—— 调整扩展屏分辨率
			
打开终端输入 xrandr 得到如下信息 Screen 0: minimum 320 x 200, current 2390 x 768, maximum 8192 x 8192 LVDS conne ...
 - ARM 平台上的Linux系统启动流程
			
开始学习嵌入式开发就一直在使用Linux系统作为学习的平台,到现在无论是PC机还是ARM开发板都已经能顺利地跑起了Linux系统,但是对Linux 的启动流程还是不甚了解.于是开始各种百度谷歌,当然看 ...
 - 深入了解float
			
1.float的历史 初衷是为了图片的文字环绕,将img设置float 2.破坏性与包裹性 a.父元素没有设置高度,内部元素浮动后,服务元素的高度被破坏了,可以将其父元素设置overflow:h ...
 - div 被Object盖住的。解决办法
			
今天遇到一个比较头疼的问题,就是在一个标签上右键,弹出的菜单div被标签内的Office控件Object挡住了下半部分,始终无法显示.查了好多解决方案,最终都不能解决问题,几乎都要放弃了.中午吃饭的时 ...
 - php返回json数据函数实例
			
本文实例讲述了php返回json数据函数的用法,分享给大家供大家参考.具体方法如下: json_encode()函数用法: ? 1 echo json_encode(array('a'=>'bb ...