Java 8 Stream:高效统计List中特定字段出现次数的教程 java list count
本教程旨在演示如何利用 Java 8 Stream API高效统计列表中特定字段的出现次数。通过将复杂的数据结构(如Listgt;)转换为POJO对象,并结合Collectors.groupingBy和Collectors.counting,可以简洁明了地实现数据与计数,极大提升代码的吸引力和维护性。
在日常的java开发中,我们经常会遇到需要对集合中的数据进行分类统计的需求。例如,给定一个包含家庭成员信息的列表,我们可能需要统计不同的数据类型成员(如兄弟姐妹、子女、婚姻)各自的数量。原始数据可能以listgt;的形式存在,其中map每个代表一个成员,键如“add_family_member”、“full_name”等。数据结构优化:从映射到 POJO
虽然Listgt;能够存储格式数据,但它缺乏类型安全,且在访问数据时需要依赖字符串键,容易出错且代码存在性差。为了更好地组织和处理数据,强烈建议使用 POJO(Plain) Old Java Object)类来表示数据实体。
为了便于家庭成员举例,我们可以定义一个 FamilyMember 类,其属性与数据字段对应:import java.time.LocalDate;public class FamilyMember { private StringmemberType; private StringfullName; private LocalDatedateOfBirth; private Stringgender; public FamilyMember(StringmemberType, StringfullName, LocalDatedateOfBirth, Stringgender) { this.memberType =memberType; this.fullName =fullName; this.dateOfBirth =dateOfBirth; this.gender =gender; } // 所有字段的 Getter public String getMemberType() { returnmemberType; } public String getFullName() { returnfullName; } public LocalDategetDateOfBirth() { returndateOfBirth; } public String getGender() { returngender; } // Setter(可选,取决于不可变性需求) public void setMemberType(StringmemberType) { this.memberType =memberType; } public void setFullName(String fullName) { this.fullName = fullName; } public void setDateOfBirth(LocalDate dateOfBirth) { this.dateOfBirth = dateOfBirth; } public void setGender(String gender) { this.gender = gender; } @Override public String toString() { return quot;FamilyMember{quot; quot;memberType='quot;memberType '\'' quot;, fullName='quot;fullName '\'' '}';
}}登录后复制
使用POJO的优点由此发现类型安全:编译时即可类型错误。代码区别性:通过点符访问属性,语义清晰。IDE操作支持:自动补全、重构等功能更强大。维护性:更容易理解和修改代码。Java 8 Stream API核心解决方案
Java 8引入的Stream API提供了一种声明式处理集合数据的方式,使得数据聚合操作变得非常简洁和高效。要统计列表中特定字段的出现次数,我们可以结合使用Collectors.groupingBy()和Collectors.counting()。
立即学习“Java免费学习笔记(深入)”;Collectors.groupingBy(Function) classifier):根据分类函数对流中的元素进行分组。分类函数的返回值将作为结果Map的键。Collectors.counting():一个下游收集器,用于计算每个分组中的元素数量。
下面是使用Java 8 Stream API统计家庭成员类型的完整示例:import java.time.LocalDate;import java.util.Arrays;import java.util.List;import java.util.Map;import java.util.stream.Collectors;public class TestFamilyMemberCount { public static void main(String[] args) { // 1.FamilyMember对象实例 创建 FamilyMember member1 = new FamilyMember(“兄弟姐妹”;,“兄弟姐妹姓名”;,LocalDate.of(1990,12,12),“男”;);FamilyMember 成员2 = new FamilyMember(“;兄弟姐妹”;,“兄弟姐妹姓名2”;,LocalDate.of(1990,12,12),“男”;);FamilyMember 成员3 = new家庭成员(“兄弟姐妹”;,“兄弟姐妹”成员3quot;, LocalDate.of(1990, 12, 12), quot;Malequot;); FamilyMembermember4 = new FamilyMember(quot;Childquot;, quot;Childnamequot;, LocalDate.of(2010, 12, 12), quot;Malequot;); FamilyMembermember5 = new FamilyMember(quot;Childquot;, quot;Childname2quot;, LocalDate.of(2000, 12, 12), quot;Femalequot;); FamilyMembermember6 = new FamilyMember(quot;Spousequot;, quot;Spousenamequot;, LocalDate.of(1990, 12, 12), quot;Malequot;); // 2. 将FamilyMember对象放入列表中 Listlt;FamilyMembergt; listOfFamilyMember = Arrays.asList(member1,member2,member3,member4,member5,member6); // 3. 使用Stream API进行备份和统计Maplt;Strin
g, Longgt; countMembers = listOfFamilyMember.stream() .collect(Collectors.groupingBy(FamilyMember::getMemberType, Collectors.counting())); // 4. 打印结果 System.out.println(countMembers); }}登录后复制
代码解析:数据准备:我们首先创建了几个FamilyMember对象实例,模拟真实的家庭成员数据。创建列表:将这些实例放入一个List中,这就是我们要处理的集合。获取流:listOfFamilyMember.strea m()将列表转换为一个Stream对象,允许我们进行链式操作。收集器:.collect()方法用于执行终止操作,将流中的元素收集到一个结果容器中。Collectors.groupingBy(FamilyMember::getMemberType, Collectors.counting())是核心部分。FamilyMember::getMemberType是一个方法引用,它作为分类函数。Stream中的每个FamilyMember对象都会调用getMemberType()方法来获取其成员类型(如“Sibling”、“Child”、“Spouse”),这个返回值将作为最终Map的键。 llectors.counting()是groupingBy的第二个参数,它是一个下游收集器。对于每个分组(即每个成员类型),counting()会计算该分组中元素的数量,这个数量最终将作为Map的值。结果:操作完成后,countMembers将是一个Map,其中按键是成员类型,值为该类型成员的数量。
输出示例:{Spouse=1, Sibling=3,Child=2}登录后复制
这表明了大多数家庭成员类型的数量。与最佳实践POJO优先事项注意事项:在处理格式化数据时,优先考虑使用POJO而非原始的Map结构。这不仅提升了代码质量,也为后续的逻辑业务处理提供了更清晰的基础。Stream API的优势:Java 8 Stream API提供了一种函数式、声明式的数据处理方式,使得代码更加简洁、易读,尤其在处理集合的过滤、映射、聚合等操作时,其表达能力远超传统的循环迭代。性能考量:Collectors.groupingBy在内部通常会创建一个HashMap来存储中间结果,其性能在大多数情况下是的。对于非常大的数据集,Stream API在多核处理器上也可以利用空闲流(parallelStream())来进一步提升性能,但需要注意空闲处理可能带来的线程安全和领头问题。通用性:教程中特定统计字段出现次数的方法具有通用性,可以根据某个属性进行排序统计的场景。只需替换FamilyMember为你的数据类,并调整getMemberType为对应的getter方法即可。
总结
通过本教程,我们学习了如何利用Java 8 Stream API中的Collectors.groupingBy和Collectors.counting,结合POJO数据模型,且高效地统计列表中特定字段的出现次数。方法这种不仅代码简洁,而且具有良好的简便性和可维护性,是现代Java开发中处理集合数据聚合的推荐实践。
以上就是Java 8 Stream:高效统计列表中特定字段出现次数的教程的详细内容,更多请关注乐哥常识网其他相关文章!