在Java开发中,我们经常会遇到这样的场景:声明两个变量指向同一个银行账户对象,修改其中一个变量的账户余额时,另一个变量查询到的余额也发生了变化。要理解这个现象,核心是搞清楚Java中对象引用的底层逻辑,以及对象和引用在内存中的存储关系。

一、先明确Java中的对象与引用的关系
Java中所有的对象都存储在堆内存中,而我们在代码中声明的变量,如果是对象类型,那么它存储的并不是对象本身,而是对象在堆内存中的地址值,这个地址值就是对象的引用。简单说,引用变量就像是一个贴着地址的便利贴,真正的数据存放在地址对应的房子里,多个便利贴可以贴同一个房子的地址。
我们可以先定义一个简单的银行账户类,作为后续分析的载体:
// 银行账户类
class BankAccount {
// 账户余额
private double balance;
public BankAccount(double initialBalance) {
this.balance = initialBalance;
}
// 存款方法
public void deposit(double amount) {
this.balance += amount;
}
// 获取余额方法
public double getBalance() {
return this.balance;
}
}
二、两个变量操作同一账户的场景复现
我们写一段测试代码,模拟两个变量指向同一个账户的情况:
public class ReferenceTest {
public static void main(String[] args) {
// 创建一个银行账户对象,初始余额1000
BankAccount account1 = new BankAccount(1000);
// 将account1的引用赋值给account2
BankAccount account2 = account1;
System.out.println("赋值后account1余额:" + account1.getBalance());
System.out.println("赋值后account2余额:" + account2.getBalance());
// 通过account2存500元
account2.deposit(500);
System.out.println("account2存款后account1余额:" + account1.getBalance());
System.out.println("account2存款后account2余额:" + account2.getBalance());
}
}
运行这段代码后,你会得到这样的输出:
赋值后account1余额:1000.0 赋值后account2余额:1000.0 account2存款后account1余额:1500.0 account2存款后account2余额:1500.0
可以看到,通过account2操作存款后,account1查询到的余额也变成了1500,这就是两个变量操作同一账户的典型表现。
三、从内存角度拆解现象原因
我们可以通过内存模型来理解整个过程:
- 第一步执行
new BankAccount(1000)时,会在堆内存中开辟一块空间存储这个账户对象,假设这块空间的地址是0x123,对象里的balance属性初始值为1000。此时account1变量存储的值就是0x123,也就是指向了这个账户对象。 - 第二步执行
BankAccount account2 = account1;时,是把account1存储的地址值0x123赋值给了account2,此时account2存储的也是0x123,两个引用变量指向了同一个堆内存中的对象。 - 第三步调用
account2.deposit(500)时,是根据account2存储的地址0x123找到对应的账户对象,修改这个对象的balance属性,balance变成1500。因为account1也指向同一个对象,所以通过account1查询到的balance自然也是1500。
四、常见误区澄清
很多人会误以为这是Java的引用传递,但实际上Java只有值传递:当传递对象是,传递的是引用的值(也就是地址值),而不是引用本身。如果我们将account2重新指向一个新的对象,比如account2 = new BankAccount(2000);,此时account2存储的地址变成了新对象的地址,后续再操作account2就不会影响account1了,这也进一步验证了两个变量操作同一对象的核心是地址值相同。
五、实际开发中的注意事项
理解这个机制后,在实际开发中要注意:如果多个变量共享同一个对象,其中一个变量的修改会影响所有指向该对象的引用,要避免无意的共享修改导致逻辑错误。如果希望两个变量操作独立的账户,一定要创建两个不同的对象,而不是简单赋值引用。如果需要复制对象,可以根据场景选择浅拷贝或深拷贝,确保引用关系的正确性。