Java方法提取优化:解决类内重复逻辑
在Java代码重构中,类内重复逻辑是影响代码质量和可维护性的常见问题。本文重点探讨如何运用方法提取技术,优化Java类中实体集合的数据转换过程,从而消除冗余代码。通过将重复的业务逻辑封装到相关实体类的新方法中,例如将角色列表转换为角色ID字符串列表的逻辑封装到`UserEntity`类的`getRoleIds()`方法中,可以显著提升代码的封装性、可读性和可维护性。本文通过具体案例,详细展示了如何识别重复代码模式,并将其高效地提取和封装,最终实现代码的精简和优化。该方法不仅减少了维护成本,还降低了潜在Bug的引入风险,是提升Java项目质量的重要实践。
在软件开发中,代码重复是一个常见的问题,它不仅增加了维护成本,还容易引入潜在的bug。当一个类中的多个方法执行相似的数据转换或集合处理逻辑时,这通常是进行重构的信号。本教程将通过一个具体的案例,演示如何利用方法提取(Method Extraction)这一重构技术,有效地解决类内代码重复问题。
识别重复代码模式
考虑以下两个Java方法,它们分别负责映射用户实体到DTO和更新用户资源:
原始方法一: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; }
在这两个方法中,以下代码片段是完全重复的:
.getRoles().stream() .map(RoleEntity::getId) .map(String::valueOf) .collect(Collectors.toList());
这段代码的作用是从 UserEntity 中获取角色列表(List
解决方案:方法提取与封装
解决这种重复代码的最佳实践是将该逻辑封装到一个独立的方法中。关键在于选择合适的方法归属。考虑到这段逻辑是关于 UserEntity 内部角色数据的转换,将其直接添加到 UserEntity 类中是更符合面向对象设计原则的做法。这不仅提高了 UserEntity 的封装性,也使得 UserEntity 自身能够更好地管理和提供其内部数据的不同表示形式。
1. 在 UserEntity 类中添加新方法
我们将提取的逻辑封装为 getRoleIds() 方法,并将其添加到 UserEntity 类中。
// UserEntity.java public class UserEntity { private Integer id; private String email; private Date lastAccessDate; private List<RoleEntity> roles; // 假设RoleEntity包含getId()方法 // ... 其他属性、构造函数、getter/setter ... /** * 获取用户所有角色的ID列表。 * 将List<RoleEntity>转换为List<String> (角色ID)。 * @return 包含角色ID字符串的列表。 */ public List<String> getRoleIds() { if (this.roles == null) { return Collections.emptyList(); // 或根据业务需求返回null } return this.roles.stream() .map(RoleEntity::getId) .map(String::valueOf) .collect(Collectors.toList()); } } // RoleEntity.java (示例) public class RoleEntity { private Integer id; private String name; // ... 其他属性、构造函数、getter/setter ... public Integer getId() { return id; } // ... }
2. 更新调用方方法
现在,原始方法中的重复代码可以被简洁地替换为对 entity.getRoleIds() 的调用。
优化后的 map 方法
protected UserDTO map(UserEntity entity) { var result = new UserDTO(); var userRoles = entity.getRoleIds(); // 直接调用UserEntity中的新方法 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())); // 确保optionalUser不为空,实际生产中应进行null检查或使用orElseThrow if (optionalUser.isPresent()) { updatedUser.setRoles(optionalUser.get().getRoleIds()); // 直接调用UserEntity中的新方法 updatedUser.setLastAccessDate(optionalUser.get().getLastAccessDate()); } else { // 处理用户不存在的情况,例如抛出异常 throw new UserNotFoundException("User with username " + updatedUser.getUserName() + " not found."); } var entity = mapToUserEntity(updatedUser); userRepository.save(entity); return updatedUser; }
注意事项与最佳实践
- 封装性提升: 将数据转换逻辑放在 UserEntity 内部,使得 UserEntity 对其自身的数据表示拥有更强的控制权,外部类无需关心其内部角色列表的具体转换细节,只需调用一个高层次的方法即可获取所需结果。
- 可读性与维护性: 新方法 getRoleIds() 具有清晰的名称,准确表达了其意图,使得调用方代码更加简洁易懂。如果未来角色ID的转换逻辑发生变化(例如,需要过滤特定类型的角色),只需修改 UserEntity 中的 getRoleIds() 方法,而无需改动所有调用它的地方。
- 选择合适的封装位置: 在本例中,将方法添加到 UserEntity 是最合适的,因为它处理的是 UserEntity 的内部数据。如果重复的逻辑不紧密依赖于某个特定实体,而是一个通用的工具函数,则可以考虑创建一个独立的工具类或私有辅助方法。
- 方法命名: 确保新方法的名称清晰、简洁,并准确反映其功能。例如,getRoleIds() 比 transformRolesToIds() 更为直接。
- 空值处理: 在 getRoleIds() 方法中,对 this.roles 进行空值检查是一个良好的实践,以避免 NullPointerException。
总结
通过方法提取和恰当的封装,我们成功地消除了类内的重复代码,提高了系统的可维护性和可读性。这个案例强调了在重构过程中,不仅要识别重复代码,更要思考将这些重复逻辑封装到何处才能最大化地提升代码质量和符合面向对象设计原则。将与特定实体数据紧密相关的逻辑放置在该实体类内部,是实现高内聚、低耦合的有效手段。
本篇关于《Java方法提取优化:解决类内重复逻辑》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于文章的相关知识,请关注golang学习网公众号!

- 上一篇
- Golang单例模式实现与使用方法

- 下一篇
- 拦截手机广告的5个实用方法
-
- 文章 · java教程 | 24分钟前 |
- RecyclerView优化:DiffUtil实现增量更新技巧
- 429浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java跨平台换行符处理技巧
- 205浏览 收藏
-
- 文章 · java教程 | 1小时前 |
- Java简单计算器实现步骤详解
- 256浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java类是什么?全面解析类的定义与结构
- 108浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java多线程技巧:高效并发实现方法
- 154浏览 收藏
-
- 文章 · java教程 | 2小时前 |
- Java获取当前秒数的两种方法
- 471浏览 收藏
-
- 文章 · java教程 | 3小时前 |
- Java反射修改final字段技巧
- 200浏览 收藏
-
- 文章 · java教程 | 4小时前 |
- 8位二进制加法实现方法详解
- 299浏览 收藏
-
- 文章 · java教程 | 4小时前 | 并发编程 同步工具
- CountDownLatch与CyclicBarrier区别对比
- 124浏览 收藏
-
- 文章 · java教程 | 4小时前 |
- Java数组高效比较:排序与二分法提速技巧
- 235浏览 收藏
-
- 文章 · java教程 | 4小时前 |
- Appium实现Gmail验证码自动验证方法
- 224浏览 收藏
-
- 文章 · java教程 | 5小时前 |
- Java实例是什么?类与实例关系全解析
- 394浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 514次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 499次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- AI Mermaid流程图
- SEO AI Mermaid 流程图工具:基于 Mermaid 语法,AI 辅助,自然语言生成流程图,提升可视化创作效率,适用于开发者、产品经理、教育工作者。
- 700次使用
-
- 搜获客【笔记生成器】
- 搜获客笔记生成器,国内首个聚焦小红书医美垂类的AI文案工具。1500万爆款文案库,行业专属算法,助您高效创作合规、引流的医美笔记,提升运营效率,引爆小红书流量!
- 711次使用
-
- iTerms
- iTerms是一款专业的一站式法律AI工作台,提供AI合同审查、AI合同起草及AI法律问答服务。通过智能问答、深度思考与联网检索,助您高效检索法律法规与司法判例,告别传统模板,实现合同一键起草与在线编辑,大幅提升法律事务处理效率。
- 733次使用
-
- TokenPony
- TokenPony是讯盟科技旗下的AI大模型聚合API平台。通过统一接口接入DeepSeek、Kimi、Qwen等主流模型,支持1024K超长上下文,实现零配置、免部署、极速响应与高性价比的AI应用开发,助力专业用户轻松构建智能服务。
- 797次使用
-
- 迅捷AIPPT
- 迅捷AIPPT是一款高效AI智能PPT生成软件,一键智能生成精美演示文稿。内置海量专业模板、多样风格,支持自定义大纲,助您轻松制作高质量PPT,大幅节省时间。
- 688次使用
-
- 提升Java功能开发效率的有力工具:微服务架构
- 2023-10-06 501浏览
-
- 掌握Java海康SDK二次开发的必备技巧
- 2023-10-01 501浏览
-
- 如何使用java实现桶排序算法
- 2023-10-03 501浏览
-
- Java开发实战经验:如何优化开发逻辑
- 2023-10-31 501浏览
-
- 如何使用Java中的Math.max()方法比较两个数的大小?
- 2023-11-18 501浏览