前言

前段时间在群里看到类似这样一个问题,下面的代码会输出什么呢?

public void test(){

String str = "hello";

change(str);

System.out.println(str);

}

private void change(String str){

str = "world";

}

当时看到这题,瞬间勾起了我的回忆。遥想当年,也曾经碰到过类似的问题,当时研究了好久才搞明白,这里再记录一下这个问题的思路。

先来说一下答案:输出:hello;

解决这类问题首先要搞明白Java到底是引用传递还是值传递。

Java到底是引用传递还是值传递

首先来解释一下什么是引用传递,什么是值传递。

引用传递(pass by reference)是指在调用方法时将实际参数的地址直接传递到方法中,那么在方法中对参数所进行的修改,将影响到实际参数。

值传递(pass by value)是指在调用方法时将实际参数拷贝一份传递到方法中,这样在方法中如果对参数进行修改,将不会影响到实际参数。

那在Java中到底是引用传递还是值传递呢?其实这个问题也一直是争论不断,而且官方也没给个确切答案。但是就我个人理解,Java是值转递。

我们先来看一个简单的例子:

public void test(){

int a = 1;

change(a);

System.out.println("a的值:" + a);

}

private void change(int a){

a = a + 1;

}

// 输出

a的值:1

在test()方法中定义了一个基本类型的变量a,然后调用change()方法试图改变这个变量,最后输出的还是原来的值。

首先我们要清楚,一个方法中的局部变量是存在栈中的,如果是基本类型的变量则直接存的是这个变量的值,如果是引用类型的变量则存的是值的地址,指向堆中具体的值。

上面的例子中,调用change()方法传递的a,其实是a变量的拷贝,不是真正的a,在change()方法中改变的是拷贝,对真正的a是没有影响的。

这么一看,Java确实是值传递,但是我们再看下面这个例子,你就会纠结了

public void test(){

User user = new User();

user.setAge(18);

change(user);

System.out.println("年龄:" + user.getAge());

}

private void change(User user){

user.setAge(19);

}

// 输出

年龄:19

看,对象里的属性被改变了,不是值传递吗,应该不会改变啊,这时候就有人总结了,当传的值是基本类型时是值传递、当传的是引用类型时是引用传递。真的是这样吗?

分析这个问题,我们需要知道变量在jvm中是怎么存储的。

首先看基本类型,这个很简单,变量在栈中直接存的是值,传到change()方法的是这个变量的拷贝,因此对拷贝的变量修改不会影响原变量的值。

接着看引用类型,变量在栈中存储的是引用地址,这个地址指向堆中具体的值,如下图:

当调用change()方法传入变量时,也是拷贝变量,但是这里的拷贝只是栈中的引用地址,并不会拷贝堆中的数据,因此会变成下图这样:

虽然变量是拷贝,但是指向的地址是同一个,因此对变量中的数据修改时,还是会影响到原来真实的变量,但是,如果我们修改的是变量在栈中的地址,则不会影响原变量,例如下面这段代码:

public void test(){

User user = new User();

user.setAge(18);

change(user);

System.out.println("年龄:" + user.getAge());

}

private void change(User user){

user = new User();

user.setAge(19);

}

// 输出

年龄:18

这种是修改变量在栈中的地址,则不会影响原变量。

说到这里,大家差不多懂了,但是回头看最开始的那个问题,传入String类型的变量,String是引用类型,按道理,原变量是会被改变的呀,结果怎么是不变呢?

String变量比较特殊,我们看String的源码可以知道,String的值是通过内部的char[]数组来维护的,但是这个数据定义的是final类型的,因此,String的值是不可变的。我们平时修改String的值,其实是重新new了一个String对象,例如下面这段代码:

String a = "hello";

a = "world";

这段代码里,其实a变量并没有被修改成world,只是重新new了一个String对象,这个对象的值是world,并把这个对象的引用地址赋给了a,原来的hello还是在堆中,只是这个值没有被引用,过段时间会被gc垃圾回收。

String变量传值在内存中的变化如下图:

String拷贝的是变量地址,但是它改变不了原String的值,因为String是不可变的,所以在change()方法中是重新new了一个String对象,改变的是新对象的值,原变量是没有影响的。

结论

Java是值传递。当传的是基本类型时,传的是值的拷贝,对拷贝变量的修改不影响原变量;当传的是引用类型时,传的是引用地址的拷贝,但是拷贝的地址和真实地址指向的都是同一个真实数据,因此可以修改原变量中的值;当传的是String类型时,虽然拷贝的也是引用地址,指向的是同一个数据,但是String的值不能被修改,因此无法修改原变量中的值。

扫一扫,关注我

java还值_Java到底是引用传递还是值传递相关推荐

  1. java string反射_Java 通过反射变更String的值过程详解

    目的 变更String对象里面的值 方法 采用反射技术,获取String里面的值,并改变 代码 通过反射技术替换掉String里面的值 package com.pojo; import java.la ...

  2. java 纯面向对象_Java到底是不是一种纯面向对象语言?

    原标题:Java到底是不是一种纯面向对象语言? Java--是否确实的 "纯面向对象"?让我们深入到Java的世界,试图来证实它. 在我刚开始学习 Java 的前面几年,我从书本里 ...

  3. java 线程分配_Java多线程原子引用分配

    我有一个缓存,该缓存是使用Simeple HashMap实现的.喜欢 - HashMap cache = new HashMap(); 大部分时间都使用此缓存从中读取值.我有另一个重新加载缓存的方法, ...

  4. java面向对象语言_Java到底是不是一种纯面向对象语言?

    英文原文:Why Java Is a Purely Object-Oriented Language Or Why Not Java是否确实是 "纯面向对象"?让我们深入到Java ...

  5. java compareto 返回值_Java comparable接口及compareTo返回值所决定的升序降序问题

    我们在学习java基础的时候知道,基本数据类型数组可以直接调动Arrays类的静态sort方法,然后输出. 例如: int iArr[] = {1,2,4,6};  Arrays.sort(iArr) ...

  6. java 函数参数 返回值_java中如何用函数返回值作为post提交的参数?

    1.我想实现的功能是在java程序中导入HttpURLConnection类,然后将函数的值作为post方法要提交的参数,最后显示在显示台上. 2.要用到的函数是自己写的可以显示实时计算机cpu.内存 ...

  7. java获取object属性值_java反射获取一个object属性值代码解析

    有些时候你明明知道这个object里面是什么,但是因为种种原因,你不能将它转化成一个对象,只是想单纯地提取出这个object里的一些东西,这个时候就需要用反射了. 假如你这个类是这样的: privat ...

  8. 接受java的返回值_java怎样接受 到return的值的?

    System.out.println(a.method(n));请解答下为什么能接收到return回来的值的?---------------------------------------packag ...

  9. java excel 取值_java实现Excel 单元格取值工具类

    在工作中经常遇到通过excel获取数据的需求,比如通过excel将数据提交到数据库等.现针对excel单元格的取值方法提取出来作为一个工具类. 具体代码如下: import org.apache.po ...

最新文章

  1. 搭建nexus后,进入首页的时候出现warning: Could not connect to Nexus.错误
  2. 【知识库专访】亲加CTO郝飞:直播技术架构解密与优化之道
  3. Java学习笔记(七)--格式化字符串及格式输出
  4. 【转】在你的博客中添加Google地图(Use Google Map API On Your Bolg)
  5. 初始化模型参数 python_pytorch 网络参数 weight bias 初始化详解_python_脚本之家
  6. Ubuntu输入ifconfig找不到IP地址,只有lo问题
  7. 调用接口时form-data形式如何传递数据
  8. CGAffineTransformMakeRotation 实现旋转
  9. ARM——操作系统—最小操作系统-开发板测试
  10. Weblogic部署
  11. 平行交通:虚实互动的智能交通管理与控制
  12. 数据库10大常见安全问题盘点
  13. java中IOException是什么异常
  14. py语言和php,php和python什么区别
  15. DYA9面向对象中--多态
  16. 随便小计:MRP元素参与计算标准配置
  17. centos开机自检及开机自启服务-Dash.shell及Systemd方法
  18. PyTorch笔记 - A ConvNet for the 2020s (ConvNeXt) 网络
  19. 奶粉事件-不合格企业及奶粉批次
  20. 如何正确测试电源的纹波?

热门文章

  1. 理解卷积神经网络的利器:9篇重要的深度学习论文(下)
  2. 开源Elasticsearch云托管服务,专享企业级服务
  3. 低代码发展专访系列之一:低代码平台产品的使用者都是谁?
  4. 云+X案例展 | 电商零售类:WakeData助力叁拾加数字化变革
  5. 程序员为什么爱穿格子衫和卫衣?
  6. 乘势而起,走进2019年风口“边缘计算”
  7. linux无法关机 grub2,Ubuntu关机卡住无法关机如何解决?
  8. mysql不支持union_MySQL中Union子句不支持order by的解决方法
  9. wepy组件子父传值_【WePY小程序框架实战三】-组件传值
  10. 设树采用孩子兄弟表示法存放.用类c语言设计算法计算树的高度.,(数据结构课程设计分类题目.doc...