- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我被禁止使用 Java 函数式编程 进行 map 构建。在我的用例中,这只是一个例子,我想构建一个 Map<Integer, Map<Integer, List<Person>>>
从 List<Person>
开始.
第一步很简单:我先构建一个 Map<Integer, List<person>>
.
然后我想添加一个新的分组级别,这样它就会生成一个新的 map :Map<Integer, Map<Integer, List<person>>>
.
重点是:如何正确使用收集器的所有功能(groupingBy、映射,也许还有其他功能)来添加新的分组级别?我所有的尝试都失败了:编译器总是告诉我我提供的所有函数都不符合它的预期。
这里是代码,简单地说:
首先是一个 Person 类,然后是一个快速构建 List 的方法:
public class Person {
private long id;
private String firstName;
private String lastName;
private Date birthDate;
private int alea;
...
此方法(位于 DataBuilder 类中)只是构建一个 Person 列表以向主类提供数据:
public List<Person> getPersons() {
List<Person> persons = new ArrayList<>();
Supplier<Person> person = Person::new;
for (int cpt = 0; cpt < 10; cpt++) {
Person p = person.get();
p.setId(cpt);
p.setFirstName("fn" + cpt);
p.setLastName("ln" + cpt);
p.setBirthDate(new Date(119, cpt, 10 + cpt));
p.setAlea(cpt % 2);
persons.add(p);
}
return persons;
}
在Main中,构建第一层Map是没有问题的:
public static void main(String[] args) {
DataBuilder db = new DataBuilder();
List<Person> persons = db.getPersons();
Map<Integer, List<Person>> mapAleaPerson = persons.stream()
.collect(Collectors.groupingBy(Person::getAlea,
Collectors.mapping(p -> p, Collectors.toList())));
mapAleaPerson.forEach((k,v) -> System.out.printf("%n%s contains %s%n", k, v));
控制台中的结果正常:
0 contains [Person [id=0, firstName=fn0, lastName=ln0, alea=0], Person [id=2, firstName=fn2, lastName=ln2, alea=0], Person [id=4, firstName=fn4, lastName=ln4, alea=0], Person [id=6, firstName=fn6, lastName=ln6, alea=0], Person [id=8, firstName=fn8, lastName=ln8, alea=0]]
1 contains [Person [id=1, firstName=fn1, lastName=ln1, alea=1], Person [id=3, firstName=fn3, lastName=ln3, alea=1], Person [id=5, firstName=fn5, lastName=ln5, alea=1], Person [id=7, firstName=fn7, lastName=ln7, alea=1], Person [id=9, firstName=fn9, lastName=ln9, alea=1]]
现在我想添加一个新的分组级别。我选择了 BirthDate 的年份,这是 Person 对象的另一个属性。
基本上,我尝试过这样的事情:
Map<Integer,Map<Integer,List<Person>>> mapByYearThenAleaPerson = persons.stream()
.collect(Collectors.groupingBy(p -> p.getBirthDate().getYear(),
Collectors.toMap(Collectors.groupingBy(Person::getAlea,
Collectors.mapping(p -> p, Collectors.toList())))));
转换没有帮助:
@SuppressWarnings("unchecked")
Map<Integer,Map<Integer,List<Person>>> mapByYearThenAleaPerson = persons.stream()
.collect(Collectors.groupingBy(p -> p.getBirthDate().getYear(),
Collectors.toMap((Function<Integer, List<Person>>) Collectors.groupingBy(Person::getAlea,
Collectors.mapping(p -> p, Collectors.toList())))));
我还尝试使用 Collectors.mapping 而不是 Collectors.toMap :
Map<Integer,Map<Integer,List<Person>>> mapByYearThenAleaPerson = persons.stream()
.collect(Collectors.groupingBy(p -> p.getBirthDate().getYear(),
Collectors.mapping((Function<Integer, List<Person>>) Collectors.groupingBy(Person::getAlea,
Collectors.mapping(p -> p, Collectors.toList())))));
每次编译器都会因为不兼容的函数类型而返回错误。
我发现添加新分组级别的唯一方法是使用旧的 Java 代码和外部循环在 Map 的 Map 中添加和分组数据。
在旧 Java 中嵌入 FP 令人失望!!
有人知道用纯 FP 来做所有这些吗?
@eran 回答后(效果很好),这是我所做的及其结果:
Map<Integer, Map<Integer, List<Person>>> mapByYearThenAleaPerson = persons.stream().collect(
Collectors.groupingBy(p -> p.getBirthDate().getYear() + 1900, Collectors.groupingBy(Person::getAlea)));
{2018={0=[Person [id=0, firstName=fn0, lastName=ln0, alea=0], Person [id=2, firstName=fn2, lastName=ln2, alea=0], Person [id=4, firstName=fn4, lastName=ln4, alea=0]], 1=[Person [id=1, firstName=fn1, lastName=ln1, alea=1], Person [id=3, firstName=fn3, lastName=ln3, alea=1]]}, 2019={0=[Person [id=6, firstName=fn6, lastName=ln6, alea=0], Person [id=8, firstName=fn8, lastName=ln8, alea=0]], 1=[Person [id=5, firstName=fn5, lastName=ln5, alea=1], Person [id=7, firstName=fn7, lastName=ln7, alea=1], Person [id=9, firstName=fn9, lastName=ln9, alea=1]]}}
真正令人惊讶的是嵌套的简单性 groupingBy
声明与构建第一个 map 本身的方式相比。
为什么我们不需要指定 mapping
和 Collectors.toList()
了?
最佳答案
使用嵌套的groupingBy
:
Map<Integer,Map<Integer,List<Person>>> mapByYearThenAleaPerson =
persons.stream()
.collect(Collectors.groupingBy(p -> p.getBirthDate().getYear(),
Collectors.groupingBy(Person::getAlea)));
关于java - 如何使用 Collectors 制作 Map of Map,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56312749/
以下面的列表为例: List input = List.of("FOO", "FOO", "FOO", "FOO", "FOO", "BAR", "BAR", "BAZ", "BAZ", "BAZ",
我有一个类似于下面的类 MyObject . public class MyObject { private String key; // not unique. multiple objec
我想获取按频率键控的数组值频率图。我能够得到相反的 - 按值键控的 map 。尝试切换参数,但 grouping by 不接受 Collector 作为第一个参数。 另一个问题,如何将 Map 实现更
我是 Java 8 Stream API 的新手,但我想用它来解决以下问题。假设我有一个名为 InputRecord 的 POJO,其中包含 name、fieldA 和 fieldB 属性,这些属性可
基准测试在 intel core i5, Ubuntu 下运行 java version "1.8.0_144" Java(TM) SE Runtime Environment (build 1.8.
我尝试使用流和收集器对值进行分组。我有我必须拆分的字符串列表。 我的数据: List stringList = new ArrayList<>(); stringList.add("Key:1,2,3
我想创建一个 Map来自 List的 Points并在映射中使用相同的 parentId 映射列表中的所有条目,例如 Map> . 我用了Collectors.toMap()但它没有编译: Map>
我已经实现了以下示例: Map> map = events.getItems().stream() .collect(Collectors.groupingBy(Event::getS
这个问题在这里已经有了答案: java 8 Collector is not a functional interface, who can tell why? (2 个回答) Java8: Usin
我想通过 Java 8 Stream 和 Collector 接口(interface)将 Map 转换为 ConcurrentHashMap,然后是我可以使用的两个选项。 第一个: Map mb =
如果我有一个对象列表(~200 个元素),只有几个唯一对象(~20 个元素)。 我只想拥有独特的值(value)。之间list.stream().collect(Collectors.toSet())
对于下面开发的 Java 8 代码,我收到以下错误。在此示例中,尝试将 Dish Name 的所有名称连接到一个变量中。使用下面的代码我得到了这个 "The method collect(Collec
我正在尝试使用rook在kubernetes集群上配置ceph,我已经运行了以下命令: kubectl apply -f common.yaml kubectl apply -f operator.y
这是我的流: Stream> futureStream = IntStream .iterate(1, n -> n n++) .mapToObj(pageNumber -> thi
Collector.of(Supplier 供应商、BiConsumer 累加器、BinaryOperator 组合器、Function 完成器、Characterstics...) Coll
按照常规思维,往一个map里put一个已经存在的key,会把原有的key对应的value值覆盖,然而通过一次线上问题,发现Java8中的Collectors.toMap反其道而行之,它默认给抛异常,
我想通过基于 LocalDateTime 的集合进行分组,但我只想获取小时,而不是分钟、秒... .collect(Collectors.groupingBy(cp -> getUpdateLocal
这个问题在这里已经有了答案: What is the point of "final class" in Java? (24 个答案) 关闭 3 年前。 为什么 Collectors 类在 Java
我有一个College具有嵌套静态类的类 Dept 学院 class College { private String collegeName; private Dept dept; public D
在Java库源代码中,Collectors#toList方法的定义如下: public static Collector> toList() { return new CollectorIm
我是一名优秀的程序员,十分优秀!