我想使用Java 8的流和lambdas将对象列表转换为Map。
这是我在Java 7及以下版本中编写它的方式。
private Map<String, Choice> nameMap(List<Choice> choices) {
final Map<String, Choice> hashMap = new HashMap<>();
for (final Choice choice : choices) {
hashMap.put(choice.getName(), choice);
}
return hashMap;
}
我可以很容易地完成这一点使用Java 8和番石榴,但我想知道如何做到这一点没有番石榴。
番石榴:
private Map<String, Choice> nameMap(List<Choice> choices) {
return Maps.uniqueIndex(choices, new Function<Choice, String>() {
@Override
public String apply(final Choice input) {
return input.getName();
}
});
}
番石榴和Java 8 lambdas。
private Map<String, Choice> nameMap(List<Choice> choices) {
return Maps.uniqueIndex(choices, Choice::getName);
}
我试图这样做,并发现,使用上面的答案,当使用Functions.identity()的键到地图,然后我有使用本地方法这样::localMethodName实际工作的问题,因为键入问题。
在这种情况下,Functions.identity()实际上对类型做了一些事情,因此该方法只能通过返回Object并接受Object的参数来工作
为了解决这个问题,我最终放弃了Functions.identity(),转而使用s->s。
所以我的代码,在我的例子中列出一个目录内的所有目录,对于每个目录,使用目录名作为映射的键,然后调用一个具有目录名的方法并返回一个项目的集合,如下所示:
Map<String, Collection<ItemType>> items = Arrays.stream(itemFilesDir.listFiles(File::isDirectory))
.map(File::getName)
.collect(Collectors.toMap(s->s, this::retrieveBrandItems));
Map<String,Choice> map=list.stream().collect(Collectors.toMap(Choice::getName, s->s));
甚至对我来说,
Map<String,Choice> map= list1.stream().collect(()-> new HashMap<String,Choice>(),
(r,s) -> r.put(s.getString(),s),(r,s) -> r.putAll(s));
例如,如果你想将对象字段转换为映射:
例对象:
class Item{
private String code;
private String name;
public Item(String code, String name) {
this.code = code;
this.name = name;
}
//getters and setters
}
和操作convert List To Map:
List<Item> list = new ArrayList<>();
list.add(new Item("code1", "name1"));
list.add(new Item("code2", "name2"));
Map<String,String> map = list.stream()
.collect(Collectors.toMap(Item::getCode, Item::getName));
这可以通过两种方式来实现。让person作为我们将要用来演示它的类。
public class Person {
private String name;
private int age;
public String getAge() {
return age;
}
}
让persons作为要转换到地图的person列表
1.在列表中使用简单的foreach和Lambda表达式
Map<Integer,List<Person>> mapPersons = new HashMap<>();
persons.forEach(p->mapPersons.put(p.getAge(),p));
2.使用在给定列表中定义的流上的收集器。
Map<Integer,List<Person>> mapPersons =
persons.stream().collect(Collectors.groupingBy(Person::getAge));
我将写如何转换列表映射使用泛型和反转控制。只是通用方法!
也许我们有一个整数列表或对象列表。那么问题是:地图的键应该是什么?
创建接口
public interface KeyFinder<K, E> {
K getKey(E e);
}
现在使用控制反转:
static <K, E> Map<K, E> listToMap(List<E> list, KeyFinder<K, E> finder) {
return list.stream().collect(Collectors.toMap(e -> finder.getKey(e) , e -> e));
}
例如,如果我们有book对象,这个类是为映射选择键
public class BookKeyFinder implements KeyFinder<Long, Book> {
@Override
public Long getKey(Book e) {
return e.getPrice()
}
}
如果相同键名的每个新值都必须重写:
public Map < String, Choice > convertListToMap(List < Choice > choices) {
return choices.stream()
.collect(Collectors.toMap(Choice::getName,
Function.identity(),
(oldValue, newValue) - > newValue));
}
如果所有的选项都必须组合在一个列表中作为一个名称:
public Map < String, Choice > convertListToMap(List < Choice > choices) {
return choices.stream().collect(Collectors.groupingBy(Choice::getName));
}
String array[] = {"ASDFASDFASDF","AA", "BBB", "CCCC", "DD", "EEDDDAD"};
List<String> list = Arrays.asList(array);
Map<Integer, String> map = list.stream()
.collect(Collectors.toMap(s -> s.length(), s -> s, (x, y) -> {
System.out.println("Dublicate key" + x);
return x;
},()-> new TreeMap<>((s1,s2)->s2.compareTo(s1))));
System.out.println(map);
配音键 AA
{12=ASDFASDFASDF, 7=EEDDDAD, 4=CCCC, 3=BBB, 2=AA}