语境:
我想在computeIfAbsent
上使用Map
函数。但是,当我使用时,我收到编译错误
String
。我使用时没有编译错误
Integer
。String
。插图:
以下陈述是合法的:
Map<Integer, List<Long>> map = new HashMap<>();
Integer key = Integer.valueOf(0);
Long value = Long.valueOf(2);
map.computeIfAbsent(key, ArrayList::new).add(value); // No compilation error
以下陈述是非法的:
Map<String, List<Long>> map = new HashMap<>();
String key = "myKey";
Long value = Long.valueOf(2);
map.computeIfAbsent(key, ArrayList::new).add(value); // Compilation error: The type ArrayList does not define ArrayList(String) that is applicable here
以下陈述是合法的:
Map<String, List<Long>> map = new HashMap<>();
String key = "myKey";
Long value = Long.valueOf(2);
map.computeIfAbsent(key, x -> new ArrayList<>()).add(value); // No compilation error
问题:我不明白为什么当与方法参考结合时,String
作为关键是特殊的。任何的想法?
当你调用ArrayList::new
而不是x -> new ArrayList<>()
时,它等于调用x -> new ArrayList<>(x)
。
方法computeIfAbsent
需要lambda表达式,其中一个lambda参数作为第二个输入参数,或者对使用String
类型的一个参数的方法的引用。
你的错误
编译错误:类型ArrayList未定义此处适用的ArrayList(String)
正在谈论:you trying to call constructor with one String argument
。因为,正如我上面所说,lambda x -> someObject.method(x)
等于someObject::method
。或者lambda x -> new SomeClass(x)
等于SomeClass::new
。
你不能在这里使用方法(构造函数)引用,因为这里需要使用一个参数或一个lambda表达式的方法(构造函数)。如果有lambda没有任何参数,你将能够调用空构造函数。