hadoop reduce阶段 对象重用问题

问题根源:对象重用

Hadoop 为了优化性能,在 reduce 方法中会重用 keyvalue 对象。

这意味着,在 for(Person p : values) 循环中,变量 p 始终指向同一个 Person 对象实例。

框架在每次迭代时,会用新的数据覆盖 这个 Person 对象的内部属性,而不是创建一个新的对象。

因此,当你执行 plist.add(p) 时,你只是将同一个对象的引用 多次添加到了列表 plist 中。

最终,plist 里所有的元素都指向内存中的同一个 Person 对象,而这个对象的值是最后一次迭代时被覆盖的结果。

错误代码示例

假设 values 中包含三个 Person 对象,其 name 属性分别为 "Alice"、"Bob" 和 "Charlie"。

复制代码
List<Person> plist = new ArrayList<>();
for(Person p : values){
    plist.add(p);
}

执行后,plist 的内容将不是 ["Alice", "Bob", "Charlie"]。相反,plist 中会包含三个指向同一个对象 的引用,而这个对象的 name 属性是 "Charlie" 。所以,遍历 plist 会得到 ["Charlie", "Charlie", "Charlie"]

正确的解决方案

你需要创建一个新的 Person 对象副本,然后将副本添加到列表中。这样,列表中的每个元素都是一个独立的对象。

方法一:手动创建新对象并复制属性

这是最直接的方法,适用于所有情况。

复制代码
List<Person> plist = new ArrayList<>();
for(Person p : values){
    // 创建一个新的Person对象
    Person newPerson = new Person();
    // 手动复制所有需要的属性
    newPerson.setName(p.getName());
    newPerson.setAge(p.getAge());
    // ... 复制其他属性
    // 将新对象的引用添加到列表中
    plist.add(newPerson);
}
方法二:使用工具类复制属性

如果你的 Person 类有很多属性,可以使用像 Apache Commons BeanUtils 这样的工具类来简化属性复制过程。

复制代码
import org.apache.commons.beanutils.BeanUtils;

List<Person> plist = new ArrayList<>();
for(Person p : values){
    Person newPerson = new Person();
    try {
        // 自动复制所有同名同类型的属性
        BeanUtils.copyProperties(newPerson, p);
    } catch (Exception e) {
        e.printStackTrace();
    }
    plist.add(newPerson);
}
方法三:实现拷贝构造函数

在你的 Person 类中定义一个拷贝构造函数,可以使代码更简洁。

复制代码
public class Person implements Writable {
    private String name;
    private int age;

    // 默认的无参构造函数(Hadoop序列化需要)
    public Person() {}

    // 拷贝构造函数
    public Person(Person other) {
        this.name = other.name;
        this.age = other.age;
    }

    // ... 其他代码 (getter, setter, write, readFields)
}

然后在 reduce 方法中这样使用:

复制代码
List<Person> plist = new ArrayList<>();
for(Person p : values){
    // 使用拷贝构造函数创建副本
    plist.add(new Person(p));
}
相关推荐
小黄人软件1 分钟前
AI时代什么是高价值目标?
大数据·人工智能
海兰2 分钟前
使用 ES|QL 调试 LLM 延迟、成本与 GPU 饱和度
大数据·elasticsearch·jenkins
财经资讯数据_灵砚智能4 分钟前
基于全球经济类多源新闻的NLP情感分析与数据可视化(夜间-次晨)2026年5月21日
大数据·人工智能·python·信息可视化·自然语言处理
Elastic 中国社区官方博客4 分钟前
用于调试 LLM 延迟、成本和 GPU 饱和度的 ES|QL 查询
大数据·人工智能·elasticsearch·搜索引擎·ai·云原生·serverless
逻极4 分钟前
RabbitMQ 从入门到精通:构建高可用、高性能的消息中间件系统
分布式·rabbitmq·消息中间件
Lyyaoo.4 分钟前
Kafka快速入门
分布式·kafka
weixin_423533997 分钟前
windows11安装claude code模型用deepseek,跳过国内校验。
大数据·elasticsearch·搜索引擎
GIS数据转换器9 分钟前
基于低空巡检的空地一体智慧治理平台
大数据·人工智能·数据挖掘·数据分析·无人机
懂AI的老郑24 分钟前
OpenClaw:高效管理分布式Agent开发团队
分布式·ai编程
来自星星的谢广坤25 分钟前
OpenClaw做分布式合适吗?
分布式·openclaw