我已经将Checker Framework v2.1.6引入java8项目并修复了所有的nullness错误,但我遇到了以下错误
Main.java:52: error: [assignment.type.incompatible] incompatible types in assignment.
List<String> collectedStrings = strings.stream().collect(Collectors.toList());
^
found : @Initialized @NonNull List<@Initialized @Nullable Object>
required: @UnknownInitialization @Nullable List<@Initialized @NonNull String>
以下(简化)示例代码抛出错误
List<String> strings = new ArrayList<>();
strings.add("test");
List<String> collectedStrings = strings.stream().collect(Collectors.toList());
collectedStrings.forEach(System.out::println);
现在我可以通过制作结果@Nullable
来解决这个问题List<@Nullable String> collectedStrings = strings.stream().collect(Collectors.toList());
但是这只是将错误级联到下一个调用,现在认为collectStrings是可疑的
有没有人为此做好工作?
答案 0 :(得分:4)
问题是Checker Framework假定Collectors.toList()
返回List<@Nullable...>
。这是一个安全,保守的假设,但在您的上下文中,您需要List<@NonNull...>
。 Checker Framework的类型推断目前太弱,无法推断出您想要的类型。
解决问题的最简单方法是将Collectors.toList()
更改为Collectors.<String>toList()
。 (只写String
等同于@NonNull String
。)
另一种方法是禁止警告,在作业上写@SuppressWarnings("nullness")
。
以下代码显示了所有这些可能性。
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.checkerframework.checker.nullness.qual.Nullable;
public class CollectorsToList {
void m(List<String> strings) {
Stream<String> s = strings.stream();
// This works:
List<String> collectedStrings1 = s.collect(Collectors.<String>toList());
// This works:
List<@Nullable String> collectedStrings2 = s.collect(Collectors.toList());
// This works:
@SuppressWarnings("nullness")
List<String> collectedStrings3 = s.collect(Collectors.toList());
// This assignment issues a warning due to incompatible types:
List<String> collectedStrings = s.collect(Collectors.toList());
collectedStrings.forEach(System.out::println);
}
}