在 Java 的代码开发过程中,为了尽可能提高方法的复用性,明确方法的作用,同时防止一个方法内部过于臃肿的问题,往往会创建许多方法,那么不可避免地会涉及到参数传递的问题。通常来说,我们将 Java 中的参数传递分为两种:值传递和引用传递。

  • 值传递:参数在进入方法时,将入参深度复制一个副本,在方法内部操作的是入参的副本,在方法执行完毕之后,外部的入参没有发生任何变化。
  • 引用传递:在方法内部操作的是参数本身,对入参做出的修改会保留到方法的外部。

  那么在 Java 中,哪些情况属于值传递,哪些情况属于引用传递呢?

1.      入参的类型

  有一种错误的见解被广为流传:如果入参是基本类型,属于值传递;如果入参不是基本类型,则属于引用传递。或者说,再深入探讨一点,和入参存储的位置相关。(基本类型存储在堆栈中,对象存储在堆中)

  以上这种说法其实并不完全正确。前半句基本认同,但是对于后半句,我可以很轻松地找到如下反例:

  由上图可知,字符串 String 类型产生的对象,是存储在堆中的非基本类型。根据以上看法,这种参数传递方式应该是引用传递,那么对字符串做出的修改应该会保存到 change(String) 方法之外,然而最终的输出结果并不是这样。把 String 改成 StringBuffer,做类似的操作,得出的结果也和 String 一致的。

  结论:基本类型的参数传递,一定是值传递;但是非基本类型的参数,其传递方式不一定是引用传递,需要进一步地分析。

2.      方法的返回类型

  另外有一种错误的看法,参数传递方式,和方法是否拥有返回值有关,如果一个方法有返回值,那么参数一定是按照值传递的。看一下如下的例子:

     public static void main(String[] args) {
Person p1 = new Person();
p1.setAge(20);
p1.setGender(0);
p1.setName("哈哈");
Person p2 = new Person();
BeanUtil.copySameFieldsObject(p1, p2);
change1(p1);
System.out.println(p1);
change2(p2);
System.out.println(p2);
} private static void change1(Person p) {
p.setAge(30);
p.setGender(1);
p.setName("呵呵");
} private static Person change2(Person p) {
p.setAge(30);
p.setGender(1);
p.setName("呵呵");
return null;
}

Test1

  如果说参数传递的方式,和方法的返回值有关,那么以上的两次输出结果一定是不同的(Person 类已经重写了 toString() 方法),因为根据以上推论,第一种方法是引用传递,第二种方法是值传递,但是实际上两次的输出结果是相同的。

  结论:参数传递方式,与方法是否有返回值,返回值的类型没有关系。

3.      真正决定入参传递方式的因素

  对于非基本类型的入参,其参数传递的方式是不定的。可以看一下如下例子:

     public static void main(String[] args) {
Person p1 = new Person();
p1.setAge(20);
p1.setGender(0);
p1.setName("哈哈");
Person p2 = new Person();
BeanUtil.copySameFieldsObject(p1, p2);
Person p3 = new Person();
BeanUtil.copySameFieldsObject(p1, p3);
change1(p1);
System.out.println(p1);
change2(p2);
System.out.println(p2);
change3(p3);
System.out.println(p3);
} private static void change1(Person p) {
p.setAge(30);
p.setGender(1);
p.setName("呵呵");
} private static void change2(Person p) {
p = new Person("呵呵", 1, 30);
} private static void change3(Person p) {
p.setName("呵呵");
p = new Person("呵呵", 1, 30);
}

Test2

  以上程序中, BeanUtil.copySameFieldsObject() 方法的作用是深度复制一份第一个参数的内容,给第二个参数,输出结果如下图所示:

通过以上研究可以得出如下结论:

  • 可以认为非基本类型的入参的参数传递方式为引用传递,但是根据方法内部执行的代码,这种传递方式存在变数,可能被转化为值传递。
  • 可以认为,方法的入参是一个对象的引用,记为 p,存放在堆栈中,这个引用指向堆中的一片内存,记为 q。当整个方法只会修改 q 的内容,而 p 始终指向 q 时,可以把整个参数传递,作为引用传递来看待。
  • 当方法内部,如果代码企图将 p 指向另一片内存 t,这时,JVM 会创建另一个引用 r,让 r 指向 t,而 p 仍然指向 q。
  • 这种试图更改指针指向的行为,主要是创建一个新的对象。创建对象的具体方法,见上一章节的内容。还包括几种特殊形式,如使用操作符“=”,创建 String 类型的对象。另外需要额外注意的是某些方法,内部实现使用了 new 创建对象,如 String.concat(String) 方法。
  • 基本类型的8种包装类型,可以当做基本类型处理,其参数传递方式虽然是引用传递,但是可以认为与值传递等价。因为这8种包装类型和 String 类型相同,其内部的数据是不可变的,这意味着任何的变动,本质上是在内存中开辟了一个新的对象。

第002弹:Java 中的值传递和引用传递的更多相关文章

  1. Does Java pass by reference or pass by value?(Java是值传递还是引用传递) - 总结

    这个话题一直是Java程序员的一个热议话题,争论不断,但是不论是你百度搜也好还是去看官方的文档中所标明的也好,得到的都只有一个结论:Java只有值传递. 在这里就不贴代码细致解释了,让我们来看看一些论 ...

  2. Java 中的值传递和参数传递

    Java中没有指针,所以也没有引用传递了,仅仅有值传递不过可以通过对象的方式来实现引用传递 类似java没有多继承 但可以用多次implements 接口实现多继承的功能 值传递:方法调用时,实际参数 ...

  3. Java中值传递和引用传递的概念

    很多书中都提到了在Java中只存在值传递,但是今天在一个NanoHTTPD的源码中看到这样一段: if (qmi >= 0) { decodeParms(uri.substring(qmi + ...

  4. Java中引用类型变量,对象,值类型,值传递,引用传递 区别与定义

    一.Java中什么叫做引用类型变量?引用:就是按内存地址查询       比如:String s = new String();这个其实是在栈内存里分配一块内存空间为s,在堆内存里new了一个Stri ...

  5. java中值传递和引用传递

    最近工作中使用到了值传递和引用传递,但是有点懵,现在看了下面的文章后清晰多了.一下是文章(网摘) 1:按值传递是什么 指的是在方法调用时,传递的参数是按值的拷贝传递.示例如下: public clas ...

  6. Java中的值传递和引用传递

    这几天一直再纠结这个问题,今天看了这篇文章有点思路了,这跟C++里函数参数为引用.指针还是有很大区别. 当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里 ...

  7. java中方法的参数传递机制(值传递还是引用传递)

    看到一个java面试题: 问:当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并可返回变化后的结果,那么这里到底是值传递还是引用传递?  答:是值传递.Java 编程语言只有值传递参 ...

  8. Java中传参的值传递和引用传递问题(转)

    今天遇到了一个java程序,需要用参数来返回值(虽然最后用另一种方法实现了),在网上看到这样一篇文章,很受启发. 本文章来自于http://hi.baidu.com/xzhilie/blog/item ...

  9. Java中的值传递

    1.先比较下字符串的比较 == 代表全等于 值和地址(存放地址) 全部相等于. equals 值等于== 和 equals的区别 列如下面的 如果name1==name2是等于的 然而name1==n ...

随机推荐

  1. 【Python图像特征的音乐序列生成】第一阶段的任务分配

    从即日起到7月20号,项目成员进行了第一次任务分配. 赵同学A.岳同学.周同学,负责了图像数据的情感数据集制作,他们根据自己的经验,对图像进行了情绪提取. 赵同学B全权负责向量映射这一块的网络搭建. ...

  2. Android(java)学习笔记137:ListView编写步骤(重点)

    1.ListView在我们的手机android编写程序中使用是十分广泛的,比如如下图中 短信 和 手机设置 都是ListView的效果: 手机设置:             短信:    2.正因为这 ...

  3. 解除phpMyAdmin导入大型MySQL数据库文件大小限制

    phpMyAdmin 导入大型数据库文件大小限制配置… 1. 修改 php.ini 文件中下列3项的值: upload_max_filesize, memory_limit 和 post_max_si ...

  4. CentOS安装RabbitMQ步骤

    1.安装gcc yum install gcc 安装 ncurses-devel yum install ncurses-devel 2.安装erlang 下载安装包 http://www.erlan ...

  5. PSNR

    PSNR,峰值信噪比,通常用来评价一幅图像压缩后和原图像相比质量的好坏,当然,压缩后图像一定会比原图像质量差的,所以就用这样一个评价指标来规定标准了.PSNR越高,压缩后失真越小.这里主要定义了两个值 ...

  6. Eclipse:Win10中设置Courier New字体

    问题:在Eclipse中设置字体的时候,没有找到Courier New字体.系统为Win10. 解决:Eclipse使用的字体为系统字体.在系统字体中有一部分是隐藏的.Courier New已经在系统 ...

  7. 正确适配苹果ATS审核要求的姿势

    首先,ATS的技术行为不会有任何变化(除了新增两个字段NSAllowsArbitraryLoadsInWebContent和NSRequiresCertificateTransparency,也就是更 ...

  8. MySQL数据库的多种备份与多种还原

    一.备份 1.mysqldump 方法备份 mysqldump备份很简单,格式如下: mysqldump -u用户名 -p密码 数据库名> XX.sql 路径 例如: mysqldump -ur ...

  9. day2-python 登录

    # username = 'niuhanyang' # 写一个判断登录的程序: # 输入: username # password # 最大错误次数是3次,输入3次都没有登录成功,提示错误次数达到上限 ...

  10. 《Spring源码深度解析》第三章 默认标签的解析

    上一章提到了,默认标签和自定义标签要分开解析.本章重点介绍默认标签的解析.在 DefaultBeanDefinitionDocumentReader 中: private void parseDefa ...