方法参数的 “值传递骗局”:修改引用参数为何不改变原对象?

一、Bug 场景

在 Java 编程中,开发人员通常认为当向方法传递一个对象引用作为参数时,在方法内部对该引用所指向对象的修改会反映到方法外部的原对象上。然而,在某些情况下,他们发现即使在方法内部看似修改了对象,原对象却并未改变,这导致了业务逻辑出现错误,程序的行为不符合预期。

二、代码示例

数据类

java 复制代码
public class Data {
    private int value;

    public Data(int value) {
        this.value = value;
    }

    public int getValue() {
        return value;
    }

    public void setValue(int value) {
        this.value = value;
    }
}

方法调用类(有缺陷)

java 复制代码
public class MethodParameterBugExample {
    public static void modifyData(Data data) {
        // 创建一个新的Data对象并赋值给参数data
        data = new Data(20); 
    }

    public static void main(String[] args) {
        Data originalData = new Data(10);
        modifyData(originalData);
        System.out.println("原对象的值: " + originalData.getValue());
    }
}

三、问题描述

  1. 预期行为 :开发人员预期在调用 modifyData 方法后,originalDatavalue 值会被修改为 20。因为他们认为传递给 modifyData 方法的 data 参数是 originalData 的引用,在方法内部对 data 的修改应该会影响到 originalData
  2. 实际行为originalDatavalue 值仍然为 10。这是因为 Java 采用的是值传递,当将 originalData 作为参数传递给 modifyData 方法时,实际上传递的是 originalData 引用的副本(即一个指向 originalData 对象的指针的拷贝)。在 modifyData 方法内部,data = new Data(20); 这行代码创建了一个新的 Data 对象,并将 data 这个引用副本指向了新对象,但原对象 originalData 的引用并未改变,它仍然指向原来的对象,所以原对象的值没有被修改。

四、解决方案

  1. 正确修改对象属性:如果要修改原对象的值,应该直接修改对象的属性,而不是重新赋值引用。
java 复制代码
public class MethodParameterBugExample {
    public static void modifyData(Data data) {
        data.setValue(20); 
    }

    public static void main(String[] args) {
        Data originalData = new Data(10);
        modifyData(originalData);
        System.out.println("原对象的值: " + originalData.getValue());
    }
}
  1. 返回修改后的对象:如果确实需要创建新对象并更新原引用,可以通过方法返回新对象,并在调用处重新赋值。
java 复制代码
public class MethodParameterBugExample {
    public static Data modifyData(Data data) {
        return new Data(20); 
    }

    public static void main(String[] args) {
        Data originalData = new Data(10);
        originalData = modifyData(originalData);
        System.out.println("原对象的值: " + originalData.getValue());
    }
}
相关推荐
华仔啊15 小时前
挖到了 1 个 Java 小特性:var,用完就回不去了
java·后端
SimonKing15 小时前
SpringBoot整合秘笈:让Mybatis用上Calcite,实现统一SQL查询
java·后端·程序员
日月云棠1 天前
各版本JDK对比:JDK 25 特性详解
java
用户8307196840821 天前
Spring Boot 项目中日期处理的最佳实践
java·spring boot
JavaGuide1 天前
Claude Opus 4.6 真的用不起了!我换成了国产 M2.5,实测真香!!
java·spring·ai·claude code
IT探险家1 天前
Java 基本数据类型:8 种原始类型 + 数组 + 6 个新手必踩的坑
java
花花无缺1 天前
搞懂new 关键字(构造函数)和 .builder() 模式(建造者模式)创建对象
java
用户908324602731 天前
Spring Boot + MyBatis-Plus 多租户实战:从数据隔离到权限控制的完整方案
java·后端
桦说编程1 天前
实战分析 ConcurrentHashMap.computeIfAbsent 的锁冲突问题
java·后端·性能优化
程序员清风2 天前
用了三年AI,我总结出高效使用AI的3个习惯!
java·后端·面试