我很了解 Java,但是,前段时间使用过它。我遇到了一些新的东西,比如
打印。<String, String>toSysOut() .withLabel("source-stream")
Esp,突出显示部分,似乎 <String, String> 是 toSysOut() 的返回类型。大多数时候,我看到诸如
Printed.toSysOut().withLabel("source-stream")
什么样的表示法是,返回类型与引用运算符 (.)
Java 有一种叫做“类型参数”的东西。他们可以使用另一种类型参数化一个类型。例如:
List<String> iCanOnlyContainStrings = new ArrayList<String>();
iCanOnlyContainStrings.add(5); // does not compile!
Run Code Online (Sandbox Code Playgroud)
方法也可以参数化。例如:
public <T> T firstNonNull(T... inputs) {
for (T input : inputs) if (input != null) return input;
}
Run Code Online (Sandbox Code Playgroud)
这个方法声明有一些我们将称之为 T 的类型。我们对此一无所知,但是这个方法的参数类型和它的返回类型都是相同的类型。因此,您可以按如下方式调用它:
String a = firstNonNull("b", "c");
Run Code Online (Sandbox Code Playgroud)
这只是编译。你是否写过:
public Object firstNonNull(Object... inputs) {
for (Object input : inputs) if (input != null) return input;
}
Run Code Online (Sandbox Code Playgroud)
然后你需要转换firstNonNull方法的返回值:(String) firstNonNull(...)。
通常java会在这里推断出你想要的类型。但是如果你想明确说明它,你可以这样做,这就是你在这里看到的语法:
public class Example {
public static <T> T firstNonNull(T... inputs) {
for (T input : inputs) if (input != null) return input;
}
public static void invokeExample() {
Serializable s = Example.<Serializable>firstNonNull(null, null);
}
}
Run Code Online (Sandbox Code Playgroud)
你必须有'.' 是明确的,这有时意味着你必须诉诸写作this.foo()或YourType.foo()只是为了你有一个点。
另一种编写代码的方法是:
Printer<String, String> printer = Printed.toSysOut();
Run Code Online (Sandbox Code Playgroud)
现在 java 的推断将正确推断方法 'toSysOut' 上的类型变量是String和String。这种语法在这些链式构建器类型场景中最常见,因为 java 不会无休止地向前推论。例如,这不起作用:
List<String> names = ImmutableList.builder().add("Hello").add("World").build();
Run Code Online (Sandbox Code Playgroud)
因为 builder() 方法是参数化的,而且 java 不会检查所有的 add 方法只是为了知道你可能想要什么(也不提前查看 build() 方法,然后回到变量声明)。因此,您会看到很多:
List<String> names = ImmutableList.<String>builder().add(...).build();
Run Code Online (Sandbox Code Playgroud)