您当前的位置:首页 > 新闻 > 科技

难住了同事:Java 方法调用到底是传值还是传引用

时间:2020-03-11 16:01:57  来源:  作者:

JAVA 方法调用中的参数是值传递还是引用传递呢?相信每个做开发的同学都碰到过传这个问题,不光是做 Java 的同学,用 C#、Python 开发的同学同样肯定遇到过这个问题,而且很有可能不止一次。

那么,Java 中到底是值传递还是引用传递呢,答案是值传递,Java 中没有引用传递这个概念。

数据类型和内存分配

Java 中有可以概括为两大类数据类型,一类是基本类型,另一类是引用类型。

基本类型

byte、short、int、long、float、double、char、boolean 是 Java 中的八种基本类型。基本类型的内存分配在栈上完成,也就是 JVM 的虚拟机栈。也就是说,当你使用如下语句时:

int i = 89;

会在虚拟机栈上分配 4 个字节的空间出来存放。

引用类型

引用类型有类、接口、数组以及 null 。我们平时熟悉的各种自定义的实体类啊就在这个范畴里。

当我们定义一个对象并且使用 new 关键字来实例化对象时。

User user = new User();

会经历如下三个步骤:

1、声明一个引用变量 user,在虚拟机栈上分配空间;

2、使用 new 关键字创建对象实例,在堆上分配空间存放对象内的属性信息;

3、将堆上的对象链接到 user 变量上,所以栈上存储的实际上就是存的对象在堆上的地址信息;

数组对象也是一样的,栈上只是存了一个地址,指向堆上实际分配的数组空间,实际的值是存在堆上的。

为了清楚的展示空间分配,我画了一张类型空间分配的示例图。

难住了同事:Java 方法调用到底是传值还是传引用

 

没有争议的基本类型

当我们将 8 种基本类型作为方法参数传递时,没有争议,传的是什么(也就是实参),方法中接收的就是什么(也就是形参)。传递过去的是 1 ,那接到的就是1,传过去的是 true,接收到的也就是 true。

看下面这个例子,将变量 oldIntValue 传给 changeIntValue 方法,该方法内对参数值进行修改,最后输出的结果还是 1。

public static void main( String[] args ) throws Exception{
    int oldIntValue = 1;
    System.out.println( oldIntValue );
    passByValueOrRef.changeIntValue( oldIntValue );
    System.out.println( oldIntValue );
}

public static void changeIntValue( int oldValue ){
    int newValue = 100;
    oldValue = newValue;
}

改变参数值并不会改变原变量的值,没错吧,Java 是按值传递。

数组和类

数组

有的同学说那不对呀,你看我下面这段代码,就不是这样。

public static void main( String[] args ) throws Exception{
    int[] oldArray = new int[] { 1, 2 };
    System.out.println( oldArray[0] );
    changeArrayValue( oldArray );
    System.out.println( oldArray[0] );
}

public static void changeArrayValue( int[] newArray ){
    newArray[0] = 100;
}

这段代码的输出是

1
100

说明调用 changeArrayValue 方法时,修改传过来的数组参数中的第一项后,原变量的内容改变了,那这怎么是值传递呢。

别急,看看下面这张图,展示了数组在 JVM 中的内存分配示例图。

难住了同事:Java 方法调用到底是传值还是传引用

 

实际上可以理解为 changeArrayValue 方法接收的参数是原变量 oldArray 的副本拷贝,只不过数组引用中存的只是指向堆中数组空间的首地址而已,所以,当调用 changeArrayValue 方法后,就形成了 oldArray 和 newArray 两个变量在栈中的引用地址都指向了同一个数组地址。所以修改参数的每个元素就相当于修改了原变量的元素。

一般我们在开发过程中有很多将类实例作为参数的情况,我们抽象出来的各种对象经常在方法间传递。比如我们定义了一个用户实体类。

public class User {

    private String name;

    private int age;

    public User(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "User{" +
                "name='" + name + ''' +
                ", age=" + age +
                '}';
    }
}

比方说我们有一个原始的实体 User 类对象,将这个实体对象传给一个方法,这个方法可能会有一些逻辑处理,比如我们拿到这个用户的 name 属性,发现 name 为空,我们就给 name 属性赋予一个随机名称,例如 “用户398988”。这应该是很常见的一类场景了。

我们通常这样使用,将 user 实例当做参数传过来,处理完成后,再将它返回。

public static void main( String[] args ) throws Exception{
    User oldUser = new User( "原始姓名", 8 );
    System.out.println( oldUser.toString() );
    oldUser = changeUserValue( oldUser );
    System.out.println( oldUser.toString() );
}

public static User changeUserValue( User newUser ){
    newUser.setName( "新名字" );
    newUser.setAge( 18 );
  return newUser;
}

但有的同学说,我发现修改完成后就算不返回,原变量 oldUser 的属性也改变了,比如下面这样:

public static void main( String[] args ) throws Exception{
    User oldUser = new User( "原始姓名", 8 );
    System.out.println( oldUser.toString() );
    changeUserValue( oldUser );
    System.out.println( oldUser.toString() );
}

public static void changeUserValue( User newUser ){
    newUser.setName( "新名字" );
    newUser.setAge( 18 );
}

返回的结果都是下面这样

User{name='原始姓名', age=8}
User{name='新名字', age=18}

那这不就是引用传递吗,改了参数的属性,就改了原变量的属性。仍然来看一张图

难住了同事:Java 方法调用到底是传值还是传引用

 

实际上仍然不是引用传递,引用传递我们学习 C++ 的时候经常会用到,就是指针。而这里传递的其实是一个副本,副本中只存了指向堆空间对象实体的地址而已。我们我们修改参数 newUser 的属性间接的就是修改了原变量的属性。

有同学说,那画一张图说这样就是这样吗,你说是副本就是副本吗,我偏说就是传的引用,就是原变量,也说得通啊。

确实是说的通,如果真是引用传递,也确实是这样的效果没错。那我们就来个反例。

public static void main( String[] args ) throws Exception{
    User oldUser = new User( "原始姓名", 8 );
    System.out.println( oldUser.toString() );
    wantChangeUser( oldUser );
    System.out.println( oldUser.toString() );
}

public static void wantChangeUser( User newUser ){
    newUser = new User( "新姓名", 18 );
}

假设就是引用传递,那么 newUser 和 main 方法中的 oldUser 就是同一个引用对象,那我在 wantChangeUser 方法中重新 new 了一个 User 实体,并赋值给了 newUser,按照引用传递这个说法,我赋值给了参数也就是赋值给了原始变量,那么当完成赋值操作后,原变量 oldUser 就应该是 name = "新名字"、age=18 才对。

然后,我们运行看看输出结果:

User{name='原始姓名', age=8}
User{name='原始姓名', age=8}

结果依然是修改前的值,我们修改了 newUser ,并没有影响到原变量,显然不是引用传递。

结论

Java 中的参数传递是值传递,并且 Java 中没有引用传递这个概念。我们通常说的引用传递,一般都是从 C 语言和 C like 而来,因为它们有指针的概念。

而我们也知道,C、C++ 中需要程序员自己管理内存,而指针的使用经常会导致内存泄漏一类的问题,Java 千辛万苦的就是为了让程序员解放出来,而使用垃圾收集策略管理内存,这其中很重要的一点就是规避了指针的使用,所以在 Java 的世界中没有所谓的指针传递。



Tags:Java   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
文章目录 如何理解面向对象编程? JDK 和 JRE 有什么区别? 如何理解Java中封装,继承、多态特性? 如何理解Java中的字节码对象? 你是如何理解Java中的泛型的? 说说泛型应用...【详细内容】
2021-12-24  Tags: Java  点击:(5)  评论:(0)  加入收藏
文章目录1、Quartz1.1 引入依赖<dependency> <groupId>org.quartz-scheduler</groupId> <artifactId>quartz</artifactId> <version>2.3.2</version></dependency>...【详细内容】
2021-12-22  Tags: Java  点击:(12)  评论:(0)  加入收藏
1 前言ObjectiveSQL 是一个Java ORM 框架,它不仅是Active Record 模式在Java 中的应用,同时还针对复杂SQL 编程提供近乎完美的解决方案,使得Java 代码与SQL 语句有机的结合,改变...【详细内容】
2021-12-13  Tags: Java  点击:(14)  评论:(0)  加入收藏
本系列为 Netty 学习笔记,本篇介绍总结Java NIO 网络编程。Netty 作为一个异步的、事件驱动的网络应用程序框架,也是基于NIO的客户、服务器端的编程框架。其对 Java NIO 底层...【详细内容】
2021-12-07  Tags: Java  点击:(17)  评论:(0)  加入收藏
流为什么动不动就说 io 流? 这个“流”是什么意思呢?流这个词,也常常出现在电竞选手的领域。大家都说,哦,这个队伍经常上去卖人头来取得局面优势的这种打法,叫献祭流。而到了 Java...【详细内容】
2021-11-15  Tags: Java  点击:(33)  评论:(0)  加入收藏
1. 字符串有整型的相互转换String a = String.valueOf(2); //integer to numeric stringint i = Integer.parseInt(a); //numeric string to an int 2. 向文件末尾添加内容B...【详细内容】
2021-10-13  Tags: Java  点击:(92)  评论:(0)  加入收藏
负载均衡是将客户端请求访问,通过提前约定好的规则转发给各个server。其中有好几个种经典的算法,下面我们用Java实现这几种算法。 轮询算法轮询算法按顺序把每个新的连接请求...【详细内容】
2021-09-27  Tags: Java  点击:(52)  评论:(0)  加入收藏
1 背景近日在给公司同事分享Arthas 工具使用时候,被它强悍的功能震撼到了就好奇研究了下它的原理及底层实现,其实它是通过Java agent 来实现的,也就深入地学习了一下Java agent...【详细内容】
2021-09-09  Tags: Java  点击:(67)  评论:(0)  加入收藏
近日浏览网上一些图片提取文字的网站,觉得甚是有趣,花费半日也做了个在线图片识别程序,完成了两个技术方案的选择,一是 tesseract + Python flask的方案实现,二是 tesseract + Sp...【详细内容】
2021-09-07  Tags: Java  点击:(81)  评论:(0)  加入收藏
Java String的判空方法是Java开发中的一个很基础的方法,下面列举了一些常用的方法。 方法一:效率高,也是最常用的方法。if(s == null || s.length() <= 0) 方法二:也是常看到的...【详细内容】
2021-09-03  Tags: Java  点击:(122)  评论:(0)  加入收藏
▌简易百科推荐
非法购买公民信息、开发人脸认证规避技术&hellip;&hellip;今年年初,广东省公安厅网安部门侦破全国首例破解“青少年防沉迷系统”的新型网络犯罪案件,抓获犯罪嫌疑人13名,查处非...【详细内容】
2021-12-28    人民日报客户端  Tags:数据安全步   点击:(5)  评论:(0)  加入收藏
就在今天,腾讯方面宣布将在2022年1月31日下架企业QQ和营销QQ,其实这一消息的降临并不让笔者意外,因为早在今年的10月28日20点之后,企业QQ和营销QQ就被停止了续费服务。相信很多...【详细内容】
2021-12-27  科技探险家    Tags:企业QQ   点击:(21)  评论:(0)  加入收藏
日前,上海交通大学发布《全球电竞之都评价报告》,对全球15个致力于发展电竞之都的城市进行评价,上海作为中国城市电竞发展的排头兵,其拥有众多优质电竞企业及完整产业集群,因此排...【详细内容】
2021-12-27  经济日报    Tags:电竞   点击:(3)  评论:(0)  加入收藏
为优化网络氛围环境,微博又开始整顿用户信息了。本月月初,微博官方发布公告,要求昵称中带有如“二货”“SB”“瘪三”“娘炮”等明显低俗或侮辱性词汇的用户尽快修改,否则将面临...【详细内容】
2021-12-24  运了个营    Tags:微博   点击:(10)  评论:(0)  加入收藏
昨日谷歌宣布,自2022年12月19日开始停止对OnHub的软件支持,OnHub路由器仍将提供Wi-Fi信号,但用户无法用谷歌Home应用程序管理它。无法更新Wi-Fi网络设置、添加额外的Wifi设备或...【详细内容】
2021-12-22  雷峰网    Tags:Google OnHub   点击:(5)  评论:(0)  加入收藏
IT之家 12 月 20 日消息,百度网盘青春版 iOS 客户端今日晚间率先开启内测,安卓客户端将在稍后内测。使用苹果 iPhone 的IT之家小伙伴可以点此下载内测版,需要先下载 TestFlight...【详细内容】
2021-12-21  IT之家    Tags:百度网盘   点击:(10)  评论:(0)  加入收藏
对于拼车单,是接还是不接,不少网约车司机表示很矛盾。接吧,钱少事多,常常跑了个寂寞,不接吧,车多客少,挑三拣四没饭吃。 在平台大力推广拼车单之下,不少司机迫于生活压力,最终还是打...【详细内容】
2021-12-17  网约车情报分享    Tags:滴滴   点击:(9)  评论:(0)  加入收藏
蓝鲸TMT频道12月16日讯,据饿了么官方微信公众号,近日,在圆桌会上,蓝骑士与平台交流了配送安全问题。饿了么表示,线上将技术手段融入安全防护;线下将持续进行安全培训,并试点智能头...【详细内容】
2021-12-17    金融界  Tags:饿了么   点击:(24)  评论:(0)  加入收藏
开源最前线(ID:OpenSourceTop) 猿妹编译项目地址: https://github.com/restic/restic全球知名代码托管平台 GitHub 今天就重磅发布了今年的年度报告&mdash;&mdash;《2021 年度 O...【详细内容】
2021-12-17  Python部落    Tags:   点击:(9)  评论:(0)  加入收藏
新京报快讯 据中国网络视听节目服务协会网站消息,12月15日,中国网络视听节目服务协会发布了《网络短视频内容审核标准细则》(2021)。中国网络视听节目服务协会组织有关短视频平...【详细内容】
2021-12-16    新京报  Tags:短视频   点击:(11)  评论:(0)  加入收藏
最新更新
栏目热门
栏目头条