简单理解Java中的值传递和引用传递

最近在做题时发现自己不太熟悉传参过程,因此写下此文,以做记录。(几年前,在简书上写的,搬运过来噜)

在Java中,传值方式有两种,一种是值传递,另一种就是引用传递,但是从本质上来说,引用传递也是一种值传递,因为引用传递传的是对象的地址,地址也是值。

虽然我们都知道,在值传递过程中,形参的变化不会影响到实参的值;在引用传递过程中,形参的变化会改变实参的值,但是在很多时候,我们不能很好的理解这两者的区别,或者单纯的认为引用传递都会使实参的值发生变化…

首先,我们先看一下值传递:
值传递就是在传参的时候,将实参拷贝一份作为样本传给形参,这就好比,你有一个珍贵的飞机模型,你的小伙伴小明找你借飞机玩,但是你担心他会把你的飞机玩坏,你就用3D打印机,重新打印一个一毛一样的给他,随便他玩,这样你的飞机就不会受到任何影响。

下面看这个eg,我们是想交换a和b的值
在这里插入图片描述
我们预期的结果肯定是a=3,b=2,对吧,但是结果却是这样的:
在这里插入图片描述
交换失败,在这个过程中,我们将x,y的值交换了,但在swap方法运行结束后,x,y的内存都被释放了,对a,b并不影响。
在这里插入图片描述
引用传递则是传入对象的地址,我们可以通过地址访问到实参对象,从而实现对对象内容的改变。
下面以数组为例进行说明:
在这里插入图片描述
输出结果:
在这里插入图片描述
从输出结果来看,实参确实发生了变化;在传参时,我们把数组b的地址传给了a,那么a便指向了数组对象,我们就可以通过脚标访问数组元素对其值进行修改。
在这里插入图片描述
有时候,引用传递也不一定会影响到实参,下面再看一个栗子
开始之前,恶补一下知识点,

1.在Java中String属于不变类,也就是说String对象的内容不能改变(关于String类是不变类,下次再讨论),但是String类仍然存在许多修改方法,such as :replace(),concat(),等等,这里就不一一列举了,其中replace()方法返回的是一个新的字符串对象,

2.而Stringbuffer类则是针对String类不可变,而产生的一个动态字符串数组,其replace()方法,它在修改字符串时,是在原字符串的基础上修改,返回值仍是原字符串(即地址未变,内容改变)。

下面看代码
在这里插入图片描述
如下图所示,

对于textstring,一开始,text指向java,将 j 替换成 i 以后,text指向了一个新的string对象(iava),然后方法执行完,形参text的内存被释放,因此textsring仍然指向Java;(图中的1.2.3是执行过程中的顺序,3过程是释放内存);

对于textbuffer,text一开始执行Java,然后加上C之后,仍然指向原来地址的字符串,(但是内容发生了变化),最后方法执行完,text释放内存,textbuffer发生了变化。
在这里插入图片描述
最后结果为:
在这里插入图片描述
第一行的iava表明,在方法内部,替换确实成功了,text指向了新的对象iava,但是最后被释放了,所以第二行前面部分,是java而不是iava.

总的来说,值传递一定不会影响实参的值,引用传递可能会影响实参的值,有时候不能影响,这种情况下就需要自己判断。

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值