
本教程详细探讨了如何利用 java 8 stream api 优化传统循环中对列表元素的条件更新和过滤操作。文章通过分析一个具体案例,展示了使用 `foreach` 结合 `removeif` 进行原地修改,以及使用 `peek`、`filter` 和 `collect` 实现函数式转换并生成新列表的两种主要策略,旨在提升代码的简洁性和可读性。
在现代 Java 应用开发中,Java 8 引入的 Stream API 已经成为处理集合数据不可或缺的工具。它提供了一种声明式、函数式的方法来处理数据,使得代码更加简洁、易读且易于并行化。本文将通过一个具体的业务场景,深入探讨如何将传统的基于循环的列表处理逻辑,优雅地转换为使用 Stream API 的现代 Java 风格。
原始问题分析
我们首先来看一个常见的业务场景:一个 Item 对象包含一个 ItemPriceCode 列表。我们需要遍历这个列表,根据每个 ItemPriceCode 的特定属性和 Item 的制造商ID,从数据库中查询相关信息,并据此更新 ItemPriceCode 中的一个字段。最后,还需要从列表中移除那些被标记为“已删除”的 ItemPriceCode。
原始代码示例如下:
import java.util.ArrayList;import java.util.List;import java.util.Optional;import java.util.stream.Collectors;// 假设的常量和实体类final class Constants { public static final String NOT_DELETED = "NOT_DELETED"; public static final String DELETED = "DELETED";}class ManufacturerPriceCodes { private String name; public ManufacturerPriceCodes(String name) { this.name = name; } public String getName() { return name; } public void setName(String name) { this.name = name; }}class ItemPriceCode { private String priceCode; private String manufacturerPriceCode; // 待更新字段 private String recordDeleted; public ItemPriceCode(String priceCode, String recordDeleted) { this.priceCode = priceCode; this.recordDeleted = recordDeleted; } // 复制构造函数,用于函数式编程场景 public ItemPriceCode(ItemPriceCode other) { this.priceCode = other.priceCode; this.manufacturerPriceCode = other.manufacturerPriceCode; this.recordDeleted = other.recordDeleted; } // Getters and Setters public String getPriceCode() { return priceCode; } public String getManufacturerPriceCode() { return manufacturerPriceCode; } public void setManufacturerPriceCode(String manufacturerPriceCode) { this.manufacturerPriceCode = manufacturerPriceCode; } public String getRecordDeleted() { return recordDeleted; } public void setRecordDeleted(String recordDeleted) { this.recordDeleted = recordDeleted; } @Override public String toString() { return "ItemPriceCode{priceCode='" + priceCode + "', manufacturerPriceCode='" + manufacturerPriceCode + "', recordDeleted='" + recordDeleted + "'}"; }}class Item { private String manufacturerID; private List itemPriceCodes; public Item(String manufacturerID, List itemPriceCodes) { this.manufacturerID = manufacturerID; this.itemPriceCodes = itemPriceCodes; } // Getters and Setters public String getManufacturerID() { return manufacturerID; } public List getItemPriceCodes() { return itemPriceCodes; } public void setItemPriceCodes(List itemPriceCodes) { this.itemPriceCodes = itemPriceCodes; } @Override public String toString() { return "Item{manufacturerID='" + manufacturerID + "', itemPriceCodes=" + itemPriceCodes + "}"; }}// 模拟的 JPA Repositoryinterface ManufacturerPriceCodesRepository { Optional findByManufacturerIDAndPriceCodeAndRecordDeleted(String manufacturerID, String priceCode, String recordDeleted);}class MockManufacturerPriceCodesRepository implements ManufacturerPriceCodesRepository { @Override public Optional findByManufacturerIDAndPriceCodeAndRecordDeleted(String manufacturerID, String priceCode, String recordDeleted) { if (Constants.NOT_DELETED.equals(recordDeleted) && "M_ID_A".equals(manufacturerID)) { if ("PC1".equals(priceCode)) return Optional.of(new ManufacturerPriceCodes("MPC_Name_1")); if ("PC2".equals(priceCode)) return Optional.of(new ManufacturerPriceCodes("MPC_Name_2")); if ("PC3".equals(priceCode)) return Optional.of(new ManufacturerPriceCodes("MPC_Name_3")); } return Optional.empty(); }}public class ItemProcessor { private ManufacturerPriceCodesRepository manufacturerPriceCodesRepository = new MockManufacturerPriceCodesRepository(); // 注入或实例化 private Item getItemManufacturerPriceCodes(Item item) { List itemPriceCodes = item.getItemPriceCodes(); // 步骤1: 遍历并条件更新 for (ItemPriceCode ipc : itemPriceCodes) { Optional mpc = manufacturerPriceCodesRepository.findByManufacturerIDAndPriceCodeAndRecordDeleted( item.getManufacturerID(), ipc.getPriceCode(), Constants.NOT_DELETED); if (mpc.isPresent()) { ipc.setManufacturerPriceCode(mpc.get().getName()); } } // 步骤2: 移除已删除的项 item.getItemPriceCodes() .removeIf(ipc -> Constants.DELETED.equals(ipc.getRecordDeleted())); return item; } // ... main 方法或测试方法}
这段代码实现了两个主要功能:
立即学习“Java免费学习笔记(深入)”;
条件更新: 遍历 itemPriceCodes 列表,根据外部查询结果更新每个 ItemPriceCode 对象的 manufacturerPriceCode 字段。原地过滤: 使用 removeIf 方法从列表中移除符合特定条件的元素(recordDeleted 为 DELETED)。
Java 8 Stream API 优化方案
我们将探讨两种主要的 Java 8 Stream API 优化策略,它们分别侧重于原地修改和函数式转换。
Seede AI
AI 驱动的设计工具
586 查看详情
1. 策略一:使用 forEach 和 removeIf 进行原地修改
这种策略是对原始代码最直接的 Java 8 化,它保留了对现有列表元素进行原地修改的特点。List.forEach() 和 List.removeIf() 都是 Java 8 引入的默认方法,非常适合这种场景。
实现方式:
条件更新: 使用 forEach 遍历 ItemPriceCode 列表,并在 lambda 表达式内部执行数据库查询和条件更新逻辑。Optional.ifPresent() 方法在这里非常有用,它能简洁地处理查询结果存在时的操作。原地过滤: removeIf 方法本身就是 Java 8 的特性,可以直接使用,它会在遍历列表时移除满足谓词条件的元素。
public class ItemProcessorOptimized1 { private ManufacturerPriceCodesRepository manufacturerPriceCodesRepository = new MockManufacturerPriceCodesRepository(); public Item getItemManufacturerPriceCodes(Item item) { List itemPriceCodes = item.getItemPriceCodes(); // 步骤1: 使用 forEach 进行条件更新 itemPriceCodes.forEach(ipc -> manufacturerPriceCodesRepository.findByManufacturerIDAndPriceCodeAndRecordDeleted( item.getManufacturerID(), ipc.getPriceCode(), Constants.NOT_DELETED) .ifPresent(mpc -> ipc.setManufacturerPriceCode(mpc.get().getName())) ); // 步骤2: 使用 removeIf 移除已删除的项 itemPriceCodes.removeIf(ipc -> Constants.DELETED.equals(ipc.getRecordDeleted())); return item; } public static void main(String[] args) { ItemProcessorOptimized1 processor = new ItemProcessorOptimized1(); List initialCodes = new ArrayList(); initialCodes.add(new ItemPriceCode("PC1", Constants.NOT_DELETED)); initialCodes.add(new ItemPriceCode("PC2", Constants.NOT_DELETED)); initialCodes.add(new ItemPriceCode("PC_DELETED", Constants.DELETED)); initialCodes.add(new ItemPriceCode("PC3", Constants.NOT_DELETED)); initialCodes.add(new ItemPriceCode("PC_NON_EXISTENT", Constants.NOT_DELETED)); Item item = new Item("M_ID_A", initialCodes); System.out.println("原始 Item: " + item); Item processedItem = processor.getItemManufacturerPriceCodes(item); System.out.println("处理后 Item (策略一): " + processedItem); }}
优点:
代码简洁,比传统 for 循环更具可读性。直接在原列表上操作,无需创建新列表,节省内存(如果这是期望的行为)。removeIf 是 List 接口自带的优化方法,效率较高。
缺点:
forEach 内部的副作用(修改 ipc 对象)与函数式编程的“无副作用”原则略有偏离。如果需要返回一个全新的列表而不是修改原有列表,则不适用。
2. 策略二:使用 peek、filter 和 collect 进行函数式转换
这种策略更符合 Stream API 的函数式编程范式,它通过一系列操作将流中的元素转换为新的元素或新的集合
以上就是Java 8 Stream API:优化列表处理与条件更新的实践指南的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1086195.html
微信扫一扫
支付宝扫一扫