作者:小牛呼噜噜 | https://xiaoniuhululu.com

计算机内功、JAVA底层、面试相关资料等更多精彩文章在公众号「小牛呼噜噜 」

经典的问题

Java 传参是值传递还是引用传递?这个问题很基础,但是许多人都有点懵

形参&实参

首先我们得了解关于参数的几个概念

形式参数:定义函数时使用的参数,用来接收函数传入参数,比如我们写个函数,函数中的参数为形式参数

public void test(String str) { //str为形式参数
System.out.println(str);
}

实际参数:我们调用函数时,函数名后面括号中的参数称为实际参数,必须有确定的值,如下面例子所示

public static void main(String[] args) {
A a = new A();
a.test("小 明"); //"小 明"则为实际参数
}

可以发现,当调用一个有参函数的时候,会把实际参数传递给形式参数。

这种传递的过程的参数一般有2种情况值传递和引用传递。

  • 值传递:调用函数时将实际参数复制一份传递到函数中,函数内部对参数内部进行修改不会影响到实际参数,即创建副本,不会影响原生对象
  • 引用传递 :方法接收的是实际参数所引用的地址,不会创建副本,对形参的修改将影响到实参,即不创建副本,会影响原生对象

我们还得知道:在Java中有2种数据类型,其中主要有基本数据类型引用数据类型,除了8种基本数据类型以外都是引用数据类型,分别是byte,short,int,long,char,boolean,float,double

Java是值传递还是引用传递?

对于这个问题,我们先来看几个例子慢慢道来:

传参的类型:基本数据类型

public class TestBasic {
public static void main(String[] args) {
int num1 = 10;
int num2 = 20;
change(num1, num2);
System.out.println("==============");
System.out.println("num1 = " + num1);
System.out.println("num2 = " + num2);
} public static void change(int param1, int param2) {
System.out.println("param1 = " + param1);
System.out.println("param2 = " + param2);
param1 = 333;
param2 = 444;
System.out.println("after change....");
System.out.println("param1 = " + param1);
System.out.println("param2 = " + param2);
}
}

结果:

param1 = 10

param2 = 20

after change....

param1 = 333

param2 = 444

==============

num1 = 10

num2 = 20

我们可以发现,change()方法内对变量重新赋值,并未改变变量num1和num2的值,改变的只是change()方法内的num1和num2的副本。

我们需要知道,基本数据类型在内存中只有一块存储空间,分配在栈stack中。

Java传参的类型如果是基本数据类型,是值传递

传参的类型:引用数据类型

public class TestQuote {

    public static void main(String[] args) {
String str = "小明";
StringBuilder str2 = new StringBuilder("今天天气好");
change(str,str2);
System.out.println("==============");
System.out.println("str = " + str);
System.out.println("str2 = " + str2); } public static void change(String param1,StringBuilder param2) {
System.out.println("param1 = " + param1);
System.out.println("param2 = " + param2);
param1= "小张";
param2.append(",我们去钓鱼");
System.out.println("after change....");
System.out.println("param1 = " + param1);
System.out.println("param2 = " + param2);
}
}

结果:

param1 = 小明

param2 = 今天天气好

after change....

param1 = 小张

param2 = 今天天气好,我们去钓鱼

str = 小明

str2 = 今天天气好,我们去钓鱼

我们发现str变量没有改变,但是str2变量却改变了,大家是不是迷惑了:Java传参的类型如果是引用数据类型,是值传递还是引用传递



其实大家被一堆术语给忽悠了,笔者画了2张图,帮助大家理解:

before change():



after change():



在Java中,除了基本数据类型以外,其他的都是引用类型,引用类型在内存中有两块存储空间(一块在栈stack中,一块在堆heap中)

如果参数是引用类型,传递的就是实参所引用的对象在栈中地址值的拷贝,这里创建的副本是 地址的拷贝。那就有人说了,可是它值变了呀,这明明就是"引用传递"嘛?

我们可以换个角度理解,如果我们把栈地址当成,会创建栈地址副本(复制栈帧),栈地址最终并没有改变,改变的是堆内存中的值。这就好比栈地址是钥匙,我们copy了一把,它能打开保险箱。我们关心的是钥匙有没有花纹这种变化,至于打开保险箱后的钱多钱少,我们并不需要关心。

虽然调用完函数后,str2变量值(堆中的数据)改变了,但是参数是引用类型,传递的实参是 栈中地址值,这是我们关心的,拷贝的是栈中地址值,最终栈中地址值并没有改变。所以是符合值传递的定义创建副本,不会影响原生对象

可能又有人问了,那str变量值为啥没有改变呢?其实这完全是由于String类的特殊,我们知道它是不可变的final,这个时候在函数中 param1= "小张";其实会隐式创建一个新的String对象,同时堆内存中会开辟一个新的内存空间,param1指向了这个新开辟的内存空间。原地址str指向的堆内存空间中数据没有任何改变。

尾语

Java中只有值传递,始终是传值的,我们要牢记,这个是官方明确说的。我们还应该清楚,其中的缘由。

参数是基本数据类型,复制的是具体值;如果参数是引用类型,把地址当成值,复制的是地址;还有String类是一个非常特殊的类,她是不可变的。


参考资料:

《深入理解Java虚拟机:JVM高级特性与最佳实践》

https://www.cnblogs.com/ITnoteforlsy/p/12266409.html


本篇文章到这里就结束啦,很感谢你能看到最后,如果觉得文章对你有帮助,别忘记关注我!

面试题:Java中为什么只有值传递?的更多相关文章

  1. Java中到底是值传递还是引用传递?

    Java中到底是值传递还是引用传递? 我们先回顾一下基本概念 实参和形参 参数在编程语言中是执行程序需要的数据,这个数据一般保存在变量中.在Java中定义一个方法时,可以定义一些参数, 举个例子: p ...

  2. Java中真的只有值传递么?

    Java中真的只有值传递么? (本文非引战或diss,只是说出自己的理解,欢迎摆正心态观看或探讨) 回顾值传递和引用传递 关于Java是值传递还是引用传递,网上有不一样的说法. 1.基本类型或基本类型 ...

  3. 2013年6月19日星期三java中函数地址值传递

    今天代码审核时确认了一个问题,理解了java中string和stringbuffer赋值问题,看到一个帖子很好,摘录如下: 理解这两个例子需要分清实参和形参的区别,引用和对象的区别 第一个例子的内部执 ...

  4. java中的参数传递——值传递、引用传递

    参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递. 在 Java 应用程序中永远不会传递对象,而只传递对象引用.因此是按引用传递对象.Java 应用程序按引用 ...

  5. java中函数是值传递还是引用传递?

    相信有些同学跟我一样,曾经对这个问题很疑惑.在网上也看了一些别人说的观点,评论不一.有说有值传递和引用传递两种,也有说只有值传递的,这里只说下个人见解 先看一个例子 public class Test ...

  6. Java 参数传递都是值传递

    Java 参数传递都是值传递,验证代码如下 public class ParamTransferTest { public static void swap(int a, int b) { int t ...

  7. java 传参方式--值传递还是引用传递

    java 传参方式--值传递还是引用传递 参数是按值而不是按引用传递的说明 Java 应用程序有且仅有的一种参数传递机制,即按值传递.写它是为了揭穿普遍存在的一种神话,即认为 Java 应用程序按引用 ...

  8. Java的参数传递是值传递还是引用传递?

    一.前言 首先先说结论,Java中方法参数传递方式是按值传递.如果参数是基本类型,传递的是基本类型的字面量值的拷贝.如果参数是引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝. 接下来深入了解一 ...

  9. Java面向对象-方法的值传递和引用传递

    Java面向对象-方法的值传递和引用传递 0 发布时间:『 2016-08-21 14:21』  博客类别:Java核心基础  阅读(197) 评论(0) Java面向对象-方法的值传递和引用传递 方 ...

随机推荐

  1. 876. Middle of the Linked List - LeetCode

    Question 876. Middle of the Linked List Solution 题目大意:求链表的中间节点 思路:构造两个节点,遍历链接,一个每次走一步,另一个每次走两步,一个遍历完 ...

  2. MySQL8小时问题

    一.问题 获取MySQL连接,8小时内无请求自动断开连接. 二.解决 2.1 分析 MySQL服务器默认的"wait_timeout"是28800秒即8小时,意味着如果一个连接的空 ...

  3. 大白话讲Java的锁

    偏向锁 对一个对象的锁偏向于某个线程,在markword中记录线程id 下次相同的线程来,直接就可以获取锁 轻量级锁 对象的Markword记录锁地址 跟线程栈里面的锁记录Lock Record的锁地 ...

  4. Kafka消息的压缩机制

    最近在做 AWS cost saving 的事情,对于 Kafka 消息集群,计划通过压缩消息来减少消息存储所占空间,从而达到减少 cost 的目的.本文将结合源码从 Kafka 支持的消息压缩类型. ...

  5. 联发科 (MTK) sensor bring up

    MT6768平台 1.添加驱动文件 2.添加硬件配置支持 3.添加硬件配置 4.添加编译配置 5.分配空间(非必要,当代码量超过当前空间大小时将会报错,根据报错log改大小即可.) 6.兼容配置 7. ...

  6. 剖析虚幻渲染体系(15)- XR专题

    目录 15.1 本篇概述 15.1.1 本篇内容 15.1.2 XR概念 15.1.2.1 VR 15.1.2.2 AR 15.1.2.3 MR 15.1.2.4 XR 15.1.3 XR综述 15. ...

  7. 使用 KubeKey 搭建 Kubernetes/KubeSphere 环境的"心路(累)历程"

    目录 今天要干嘛? 在哪里干? 从哪里开始干? 快速开干! 解决依赖问题再继续干! 如何干翻重来? 连着 KubeSphere 一起干! 干不过,输了. 重整旗鼓,继续干! 再次重整旗鼓,继续干! 一 ...

  8. Hyperledger Fabric 智能合约开发及 fabric-sdk-go/fabric-gateway 使用示例

    前言 在上个实验 Hyperledger Fabric 多组织多排序节点部署在多个主机上 中,我们已经实现了多组织多排序节点部署在多个主机上,但到目前为止,我们所有的实验都只是研究了联盟链的网络配置方 ...

  9. camunda BPM支持的开发和运行环境

    以Camunda7.13版本为例,介绍Camunda支持的开发运行环境. 一.支持的Java开发环境 • Java版本: 8 / 9 / 10 / 11 / 12 / 13 / 14• Springb ...

  10. 『忘了再学』Shell基础 — 32、Shell中test测试命令详解

    目录 1.test测试命令 (1)test命令介绍 (2)test命令使用方式 (3)示例 2.按照文件类型进行判断 3.按照文件权限进行判断 4.两个文件之间进行比较 5.两个整数之间比较 6.字符 ...