java是引用传递还是值传递_Java是按引用传递还是按值传递?

本文深入探讨了Java中参数传递的机制,特别是对象引用按值传递的概念。通过示例说明,即使Java传递的是对象引用的副本,但只要这个副本指向堆中的对象,对象的实际值仍可被方法内部修改。对于不可变对象如String,其值不能更改,而可变对象如StringBuilder则允许在方法中改变其状态。文章还提醒读者注意原始类型的值传递和不可变对象的区别,并提供了一个挑战性的代码示例来巩固理解。
摘要由CSDN通过智能技术生成

c6d64d1d1cfe2844dc7b7c988e9c64f8.png

许多编程语言都允许按引用或按值传递参数。在Java中,我们只能按value传递参数。这施加了一些限制,并且引起了疑问。例如,如果在方法中更改了参数值,方法执行后该值会怎样?您可能还想知道Java如何管理内存堆中的对象值。该Java Challenger可帮助您解决有关Java中对象引用的这些以及其他常见问题。

获取源代码

在遵循示例的同时,您可以运行自己的测试。源码地址: https://github.com/rafadelnero/javaworld-challengers

对象引用按值传递

Java中的所有对象引用均按值传递。这意味着该值的副本将传递给方法。但是诀窍在于传递值的副本也会更改对象的实际值。要了解原因,请从以下示例开始:

public class ObjectReferenceExample {public static void main(String... doYourBest) {        Simpson simpson = new Simpson();        transformIntoHomer(simpson);        System.out.println(simpson.name);}static void transformIntoHomer(Simpson simpson) {        simpson.name = "Homer";}}class Simpson {String name;}

您认为执行simpson.name该transformIntoHomer方法后会怎样?

在这种情况下,它将是荷马!原因是Java对象变量只是指向内存堆中实际对象的引用。因此,即使Java通过值将参数传递给方法,但是如果变量指向对象引用,则实际对象也将被更改。

如果您仍然不太清楚它是如何工作的,请看下图。6216348f5da60c3cc00bec62dc32d3b8.png

拉斐尔·奇内拉托·德尔尼罗

原始类型是否按值传递?

像对象类型一样,基本类型也按值传递。您能否在下面的代码示例中推断出原始类型会发生什么?

public class PrimitiveByValueExample {public static void main(String... primitiveByValue) {        int homerAge = 30;        changeHomerAge(homerAge);        System.out.println(homerAge);}static void changeHomerAge(int homerAge) {        homerAge = 35;}}

如果您确定该值将更改为30,那么您是正确的。这是30,因为(再次)Java通过值传递对象参数。数字30只是值的副本,而不是实际值。基本类型在堆栈存储器中分配,因此仅本地值将被更改。在这种情况下,没有对象引用。

传递不可变的对象引用

如果我们对不可变的String对象进行了相同的测试该怎么办?

JDK包含许多不可变的类。实例包括包装类型Integer,Double,Float,Long,Boolean,BigDecimal,和当然的非常公知的String类。

在下一个示例中,请注意当我们更改a的值时会发生什么String。

public class StringValueChange {public static void main(String... doYourBest) {        String name = "";        changeToHomer(name);        System.out.println(name);}static void changeToHomer(String name) {        name = "Homer";}}

您认为输出结果是什么?如果您猜到了“”,那就恭喜!发生这种情况是因为String对象是不可变的,这意味着内的字段String是最终字段,无法更改。

使String类不可变使我们可以更好地控制Java最常用的对象之一。如果a的值String可以更改,则会产生很多错误。还要注意,我们没有更改String类的属性;相反,我们只是String为其分配了一个新值。在这种情况下,“荷马”值将被传递到name在changeToHomer方法。该方法完成执行后,String“本垒打”将有资格被垃圾回收changeToHomer。即使无法更改对象,局部变量也会更改。

传递可变对象引用

与不同String,JDK中的大多数对象都是可变的,例如StringBuilder类。下面的示例与上一个示例相似,但是功能StringBuilder而不是String:

static class MutableObjectReference {    public static void main(String... mutableObjectExample) {      StringBuilder name = new StringBuilder("Homer ");      addSureName(name);      System.out.println(name);    }    static void addSureName(StringBuilder name) {      name.append("Simpson");    }  }

您能否推断出此示例的输出?在这种情况下,因为我们正在使用可变对象,所以输出将为“ Homer Simpson”。您可能期望Java中的任何其他可变对象具有相同的行为。

您已经了解到Java变量是通过值传递的,这意味着将传递值的副本。只要记住复制的值指向Java内存堆中的真实对象即可。按值传递仍然会更改实际对象的值。

接受对象引用挑战!

在此Java Challenger中,我们将测试您从对象引用中学到的知识。在下面的代码示例中,您将看到不可变String和可变的StringBuilder类。每个参数都作为参数传递给方法。知道Java仅按值传递,一旦执行了此类的main方法,您认为输出是什么?

public class DragonWarriorReferenceChallenger {  public static void main(String... doYourBest) {    StringBuilder warriorProfession = new StringBuilder("Dragon ");    String warriorWeapon = "Sword ";    changeWarriorClass(warriorProfession, warriorWeapon);

System.out.println("Warrior=" + warriorProfession + " Weapon=" + warriorWeapon);  }  static void changeWarriorClass(StringBuilder warriorProfession, String weapon) {    warriorProfession.append("Knight");    weapon = "Dragon " + weapon;

weapon = null;    warriorProfession = null;  }}

这些是选项,请在本文结尾处查看答案。

A:战士=空武器=空B:战士=龙武器=龙C:战士=龙骑士武器=龙剑D:战士=龙骑士武器=剑

发生什么事了

上面示例中的第一个参数是warriorProfession变量,它是一个可变对象。第二个参数,武器是不可变的String:

static void changeWarriorClass(StringBuilder warriorProfession, String weapon) {    ...  }

现在,让我们分析一下该方法内部发生的情况。在此方法的第一行,我们将Knight值附加到warriorProfession变量。请记住,这warriorProfession是一个可变的对象;因此实际对象将被更改,并且其值将为“ Dragon Knight”。

warriorProfession.append("Knight");

在第二条指令中,不变的局部String变量将更改为“ Dragon Sword”。但是,由于实物String是不可变的并且其属性是最终的,因此实物永远不会更改:

weapon = "Dragon " + weapon;

最后,我们传递null给这里的变量,而不传递给对象。只要它们仍然可以从外部访问,这些对象将保持不变-在这种情况下,可以通过main方法访问。而且,尽管局部变量将为null,但对象将不会发生任何事情:

weapon = null;warriorProfession = null;

从所有这些我们可以得出结论,我们可变StringBuilder且不变的最终值String将是:

System.out.println("Warrior=" + warriorProfession + " Weapon=" + warriorWeapon);

changeWarriorClass方法中唯一更改的值是warriorProfession,因为它是可变StringBuilder对象。请注意,它warriorWeapon没有改变,因为它是一个不可变的String对象。

我们的挑战者代码的正确输出为:

D:战士=龙骑士武器=剑。

视频挑战!调试Java中的对象引用

调试是完全吸收编程概念并改善代码的最简单方法之一。在本视频中,您可以在调试和解释Java对象引用时进行后续操作。

视频地址:https://youtu.be/HXvKLn5RkRQ

对象引用的常见错误尝试通过引用更改不可变值。

尝试通过引用更改原始变量。

当您在方法中更改可变对象参数时,期望真实对象不会更改。

关于对象引用要记住什么Java总是按值传递参数变量。

Java中的对象变量始终指向内存堆中的实际对象。

可变对象的值传递给方法时可以更改。

不可变对象的值即使传递了新值也无法更改。

“按值传递”是指传递值的副本。

“通过引用传递”是指在内存中传递变量的真实引用。

28f2a2055a1e09b53ba3a0504fe844d5.png

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值