Java 只有值传递!为什么?

简介: 面试官爱问的一个基础问题:Java是值传递还是引用传递?想必大家都对这个问题都有自己的看法,那到底事实是怎样的,我们又该如何回答面试官这个问题呢?今天咱们就来好好分析一波

面试官爱问的一个基础问题:Java是值传递还是引用传递?


想必大家都对这个问题都有自己的看法,那到底事实是怎样的,我们又该如何回答面试官这个问题呢?今天咱们就来好好分析一波


值传递?引用传递?

首先,我们得先知道什么叫值传递,什么叫引用传递,知道这个才能理解Java到底如何做的。若想理解这两种传递需要先理解形式参数和实际参数两个概念

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

(String str) { System..println(str)}

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

(String[] args) {
    A a = A()a.test()}

可以发现,当调用一个有参函数的时候,会把实际参数传递给形式参数;于是这个传递的过程便有两种情况,即值传递和引用传递


值传递就是把参数的值给你,调用函数时将实际参数复制一份传递到函数中,这样函数内部对参数内部进行修改不会影响到实际参数;而引用传递就不一样了,它直接把参数的实际地址给调用函数了,函数内部可直接修改该地址内容,会影响到实际参数


我来举个例子,我司有一个数据库A,仅允许内部人员操作,现在有个项目需要和别的公司合作,该数据库的数据需要交给合作公司一份,我总不能直接把我司数据库A地址给他们,让他们直接连我们数据库A吧,他们要是删库跑路了,那我boss岂不要杀我祭天了


所以这个时候,把我司数据库表数据拷贝一份到一个新的数据库B,合作公司可以看这个数据库B数据,他们也可以随意操作,不会影响我司数据库


这个操作就类似于值传递,如果合作公司直接操作我司数据库,就类似于引用传递了,这下大家应该晓得两者之别了


Java值传递还是引用传递?

我们了解了值传递和引用传递的概念,那Java中到底是哪种传递方式呢?我们来看代码分析

(String[] args) {
    Fans fans = Fans()t = fans.test(t)System..println(+ t) }
(t) {
    t = System..println(+ t) }
//输出
In testIn main

上述代码,在main中定义t的值是1,在函数test中修改了参数t的值为2,这个结果是不是很容易分析出来了呢?test函数并未改变传入的t的值,那按照上面我们的介绍是不是可以得出结论:Java中是值传递

有的人可能会质疑,你传入的参数t是基本类型,你若传入引用对象类型,那肯定就会改变对象内容了,OK,再来看一段引用类型代码

= String = (Fans fans) {
    fans.= fans.= System..println(+ fans.++fans.)}
(String[] args) {
    Fans fa = Fans()fa.test(fa)System..println(+ fans.getA()++fans.getB())}
//输出
In testABnewFans
In mainABnewFans

哎啊,输出结果竟然一样了,也就是传入的fans对象被函数test修改了,那这样是不是又变成了引用传递了?

于是得出结论,基本类型是值传递,引用类型是引用传递?事实是这个样子吗,我们再来通过String这个引用类型实验下

(String t) {
    t = System..println(+ t)}
(String[] args) {
    Fans fans = Fans()String tt = fans.test(tt)System..println(+ t)}
//输出
In test:关注公众号
In main:Java技术栈

啊,这,咋肥事,传递的参数值并未修改,怎么又变成值传递了


上述三个例子表现结果各有差异,到底结论是什么呢?一起来分析下


第一个传入的是基本类型,基本类型指向的就是数值,传递类似于赋值操作,不会对原数值产生影响,就是类似于a=10,b=a,b=20这种,并不会使a变为20;


第二个引用对象传入的是引用类型fans的地址的值,传入的原参数fa指向地址0x123456,所以函数test的参数fans也指向0x123456,函数内部对引用fans进行修改,于是修改了0x123456地址的值,造成外部改变


第三个引用对象是String类型,同样传入的是原参数tt的指向地址0x123456,函数test参数t也是指向0x123456的值,那为什么这个和第二个结果不一样的嘞?重点在于 t="关注公众号"; 这一句本来想尝试着使内容"Java技术栈"改变成"关注公众号",但是无奈,String类型是static final类型的,这个大家应该晓得的不,不晓得的该去补课读读String的源码了,于是变成了 t=new String("关注公众号"),t指向了另一个地址,这个地址的内容是"关注公众号",所以原来的引用tt还是指向原来的地址0x123456,并未改变


有的同学可能会提出问题了,为什么第二个可以改变这个地址的内容,第三个不行?很明显啊,String是final的,不可修改,而第二个可以直接修改该地址的内容;那问题又来了,既然这样,还能叫值传递吗?


告诉你,就是值传递,因为我们第二个的验证方法不对,你如果在函数的第一行加上个fans = new Fans();你看看它还输出啥,这就变成和第三个String类似的道理了,改变了函数参数的指向位置,函数外部和函数内部输出就不一样了,函数内部也就不会影响外部了;如果按照应引用传递,即使加了这一句,也应该是函数内外都是输出一样的,况且,这也有悖于引用传递的会改变传入参数的概念


思考

值传递和引用传递并不是按照传递的内容来区分的,传递的是引用的并不一定的引用传递,根据定义结果来区分;


在Java中用的是值传递(记好咯,下次面试别回答错了)


在其它方法里面改变引用类型的值都是通过引用改变的,当传递引用对象的时候,传递的是复制的引用的对象句柄,是复制过的,也就是在内存中复制了一个句柄,这两个句柄指向同一个对象,所以你改变这个句柄对应的空间的数据会影响到外部的变量


虽然是复制的,但是指向的是同一个地址,当你把这个句柄指向其它对象的引用时并不会改变原来的值(例子三String),因为你用的是复制过的句柄


相关文章
|
存储 Java 程序员
Java中相等比较与值传递
Java中相等比较与值传递
79 0
|
存储 安全 Java
【面试题精讲】为什么 Java 只有值传递?
【面试题精讲】为什么 Java 只有值传递?
|
4月前
|
Java
java是值传递还是引用传递
本文澄清了Java中参数传递的常见误解,总结出Java采用“值传递”的方式。对于基本类型,传递其值的拷贝,方法内修改不影响原值;而对于对象类型,则传递其引用地址的拷贝,尽管是拷贝,但因指向同一对象,故方法内的修改会影响原对象状态。形参仅在方法内部有效,而实参则是调用方法时传递的具体值。通过示例和比喻(如复刻仓库钥匙),形象地解释了值传递、引用传递及Java特有的“共享对象传递”概念,帮助理解不同情况下参数传递的行为差异。
|
4月前
|
Java
java中的值传递和引用传递
【8月更文挑战第3天】在Java中,值传递用于基本数据类型,传递的是值的副本,因此方法内的修改不影响原值;而引用传递用于对象和数组,虽传递的是引用的副本,但对对象内容的修改会影响原始对象。理解这两者对于正确处理方法调用及参数至关重要。
|
4月前
|
Java
java中的值传递和引用传递
【8月更文挑战第2天】在Java中,基本数据类型如`int`、`double`等采用值传递,传递的是变量值的副本,因此方法内的修改不影响原变量。对象类型则通过引用传递,传递的是对象引用的副本,允许方法内修改原对象。例如,对`StringBuilder`对象的修改会影响原始对象。
|
7月前
|
存储 安全 Java
Java方法的值传递技术详解
Java方法的值传递技术详解
44 3
|
6月前
|
Java
Java的值传递与“引用传递”辨析
Java的值传递与“引用传递”辨析
30 0
|
7月前
|
JavaScript 前端开发 Java
【JAVA面试题】什么是引用传递?什么是值传递?
【JAVA面试题】什么是引用传递?什么是值传递?
|
7月前
|
Java
每日一道Java面试题:Java是值传递还是引用传递?
每日一道Java面试题:Java是值传递还是引用传递?
41 1
|
7月前
|
存储 Java
如何理解Java是按值传递
如何理解Java是按值传递