0

0

Java代码重构实践:通过方法提取消除重复逻辑

花韻仙語

花韻仙語

发布时间:2025-08-28 15:51:01

|

960人浏览过

|

来源于php中文网

原创

Java代码重构实践:通过方法提取消除重复逻辑

本教程深入探讨在Java开发中,如何通过将重复的业务逻辑封装为独立方法来有效消除代码冗余,从而提升代码的可维护性和可读性。针对跨多个方法共享的特定逻辑片段,我们将演示如何将其提取并集成到相关实体类中,例如在 UserEntity 中添加 getRoleIds() 方法,以实现代码的精简与高效复用,同时增强领域模型的表达能力。

1. 问题背景与代码冗余分析

在软件开发过程中,代码冗余是一个常见且亟待解决的问题。当同一段逻辑出现在应用程序的不同方法中时,不仅增加了代码量,更重要的是降低了代码的可维护性、可读性,并增加了引入缺陷的风险。例如,当业务规则发生变化时,开发者需要修改所有包含该重复逻辑的地方,这极易遗漏或引入不一致性。

考虑以下Java代码示例,其中 map 方法用于将 UserEntity 转换为 UserDTO,而 updateUser 方法则处理用户更新逻辑:

原始代码片段:

// 方法一:map
protected UserDTO map(UserEntity entity) {
    var result = new UserDTO();
    // 存在重复逻辑
    var userRoles = entity.getRoles().stream()
            .map(RoleEntity::getId)
            .map(String::valueOf)
            .collect(Collectors.toList());
    result.setId(entity.getId().toString());
    result.setLastAccessDate(entity.getLastAccessDate());
    result.setRoles(userRoles);
    if (entity.getEmail() != null) {
        var email = new UserDTO.Email(entity.getEmail(), EMAIL_TYPE);
        result.setEmails(List.of(email));
    }
    return result;
}

// 方法二:updateUser
public UserResource updateUser(String id, UserResource updatedUser) {
    var optionalUser = userRepository.findById(Integer.valueOf(updatedUser.getUserName()));
    // 存在重复逻辑
    updatedUser.setRoles(optionalUser.get().getRoles()
            .stream()
            .map(RoleEntity::getId)
            .map(String::valueOf)
            .collect(Collectors.toList()));
    updatedUser.setLastAccessDate(optionalUser.get().getLastAccessDate());
    var entity = mapToUserEntity(updatedUser);
    userRepository.save(entity);
    return updatedUser;
}

在这两个方法中,提取用户角色ID列表的逻辑完全相同:

.getRoles().stream()
.map(RoleEntity::getId)
.map(String::valueOf)
.collect(Collectors.toList())

这种重复不仅使得代码难以阅读,也使得未来对角色ID提取逻辑的任何修改都必须在两个地方进行,增加了出错的可能性。

立即学习Java免费学习笔记(深入)”;

2. 解决方案:提取方法(Extract Method)

解决代码冗余最有效且最常用的策略之一是“提取方法”(Extract Method)重构技术。其核心思想是将一段重复的或具有独立逻辑的代码块封装成一个新的方法,然后在原有的位置调用这个新方法。

针对上述问题,最佳实践是将提取角色ID列表的逻辑封装到 UserEntity 类自身的一个新方法中。这样做有以下几个优点:

智简简历
智简简历

免费AI简历制作工具,智能生成、可视化编辑、多格式导出。

下载
  1. 增强领域模型: UserEntity 作为领域对象,拥有获取其角色ID列表的能力是符合其职责的,这使得实体更加“智能”和自洽。
  2. 提高内聚性: 将与 UserEntity 相关的操作放在 UserEntity 类中,符合高内聚的设计原则。
  3. 简化调用: 调用方无需了解角色ID提取的具体实现细节,只需调用一个语义明确的方法即可。

2.1 改造 UserEntity 类

首先,在 UserEntity 类中添加一个名为 getRoleIds() 的新方法,用于封装提取角色ID的逻辑。

// 假设这是您的 UserEntity 类
public class UserEntity {
    private Long id;
    private String email;
    private Date lastAccessDate;
    private List<RoleEntity> roles; // 假设 RoleEntity 包含 getId() 方法

    // ... 其他属性和方法

    /**
     * 获取用户所有角色的ID列表。
     * @return 包含角色ID字符串的列表。
     */
    public List<String> getRoleIds() {
        if (this.roles == null || this.roles.isEmpty()) {
            return Collections.emptyList();
        }
        return this.roles.stream()
                .map(RoleEntity::getId)
                .map(String::valueOf)
                .collect(Collectors.toList());
    }

    // Getter和Setter方法
    public Long getId() { return id; }
    public void setId(Long id) { this.id = id; }
    public String getEmail() { return email; }
    public void setEmail(String email) { this.email = email; }
    public Date getLastAccessDate() { return lastAccessDate; }
    public void setLastAccessDate(Date lastAccessDate) { this.lastAccessDate = lastAccessDate; }
    public List<RoleEntity> getRoles() { return roles; }
    public void setRoles(List<RoleEntity> roles) { this.roles = roles; }
}

// 假设 RoleEntity 类
class RoleEntity {
    private Long id;
    private String name;

    public RoleEntity(Long id, String name) {
        this.id = id;
        this.name = name;
    }

    public Long getId() { return id; }
    public void setId(Long id) { this.id = id; }
    public String getName() { return name; }
    public void setName(String name) { this.name = name; }
}

2.2 重构调用方方法

一旦 UserEntity 拥有了 getRoleIds() 方法,我们就可以在原先存在重复逻辑的地方直接调用它,从而大大简化代码。

重构后的 map 方法:

protected UserDTO map(UserEntity entity) {
    var result = new UserDTO();
    // 调用 UserEntity 的新方法
    var userRoles = entity.getRoleIds();
    result.setId(entity.getId().toString());
    result.setLastAccessDate(entity.getLastAccessDate());
    result.setRoles(userRoles);
    if (entity.getEmail() != null) {
        var email = new UserDTO.Email(entity.getEmail(), EMAIL_TYPE);
        result.setEmails(List.of(email));
    }
    return result;
}

重构后的 updateUser 方法:

public UserResource updateUser(String id, UserResource updatedUser) {
    var optionalUser = userRepository.findById(Integer.valueOf(updatedUser.getUserName()));
    // 调用 UserEntity 的新方法
    updatedUser.setRoles(optionalUser.get().getRoleIds());
    updatedUser.setLastAccessDate(optionalUser.get().getLastAccessDate());
    var entity = mapToUserEntity(updatedUser);
    userRepository.save(entity);
    return updatedUser;
}

通过这种方式,原本冗长的流式操作被一个简洁、语义明确的方法调用所取代,代码变得更加清晰和易于维护。

3. 注意事项与最佳实践

在进行方法提取重构时,需要考虑以下几点以确保重构的质量和有效性:

  • 方法命名: 新方法的名称应清晰、准确地表达其功能和意图。例如,getRoleIds() 明确表示获取角色ID列表。
  • 方法可见性: 根据新方法的职责和调用范围,合理设置其可见性(public, protected, private)。如果它只在类内部使用,则应设为 private。在本例中,由于 getRoleIds() 可能被其他外部类(如服务层或DTO转换逻辑)调用,public 是合适的选择。
  • 参数设计: 如果被提取的逻辑需要外部数据,应将这些数据作为参数传递给新方法。本例中,getRoleIds() 直接操作 UserEntity 自身的 roles 属性,无需额外参数。
  • 无副作用原则: 尽量确保提取的方法是“纯函数”,即不修改其外部状态,只根据输入产生输出。如果必须有副作用,应明确文档说明。
  • 测试覆盖: 在进行任何重构之前,确保有充分的单元测试覆盖。重构后,运行这些测试以验证功能的正确性未受影响。
  • 粒度适中: 提取的方法不宜过大也不宜过小。过大可能包含多种职责,过小则可能引入过多方法调用开销且无明显收益。本例中提取的逻辑粒度适中,专注于一个单一的职责。

4. 总结

代码重构是软件开发中不可或缺的一部分,它有助于提升代码质量、降低维护成本。本文通过一个具体的Java代码冗余示例,演示了如何运用“提取方法”这一核心重构技术,将重复逻辑封装到领域实体类中。这种做法不仅有效地消除了代码冗余,还增强了领域模型的表达能力和代码的可读性。遵循这些最佳实践,开发者可以构建出更健壮、更易于管理和扩展的应用程序。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang map内存释放
golang map内存释放

本专题整合了golang map内存相关教程,阅读专题下面的文章了解更多相关内容。

77

2025.09.05

golang map相关教程
golang map相关教程

本专题整合了golang map相关教程,阅读专题下面的文章了解更多详细内容。

40

2025.11.16

golang map原理
golang map原理

本专题整合了golang map相关内容,阅读专题下面的文章了解更多详细内容。

67

2025.11.17

java判断map相关教程
java判断map相关教程

本专题整合了java判断map相关教程,阅读专题下面的文章了解更多详细内容。

47

2025.11.27

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

74

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

38

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

83

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

97

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

223

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 80.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号