我对Java很熟悉,但在一段时间前就使用过它。我遇到了一些新的东西,比如
已打印<字符串,字符串>toSysOut((.withLabel("源流"(
Esp,突出显示的部分,似乎<字符串,字符串>是toSysOut((的返回类型。大多数时候,我看到这样的
Printed.toSysOut((.带标签("源流"(
这是一种什么样的表示法,其中返回类型与引用运算符(.(一起使用
Java有一种叫做"类型参数"的东西。他们可以使用另一个类型来参数化一个类型。例如:
List<String> iCanOnlyContainStrings = new ArrayList<String>();
iCanOnlyContainStrings.add(5); // does not compile!
方法也可以参数化。例如:
public <T> T firstNonNull(T... inputs) {
for (T input : inputs) if (input != null) return input;
}
这个方法声明有一个我们将调用T的类型。我们对此一无所知,但这个方法的参数类型及其返回类型都是相同的类型。因此,您可以按如下方式调用它:
String a = firstNonNull("b", "c");
这只是编译。你写过吗:
public Object firstNonNull(Object... inputs) {
for (Object input : inputs) if (input != null) return input;
}
然后,您需要强制转换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);
}
}
你必须有"。"为了显式,这有时意味着你必须求助于写this.foo()
或YourType.foo()
,这样你就有了一个点。
另一种编写代码的方法是:
Printer<String, String> printer = Printed.toSysOut();
现在,java的推断将正确地推断出方法"toSysOut"上的类型变量是String
和String
。这种语法在这些链式构建器类型的场景中最常见,因为java不会无休止地向前看以进行推断。例如,这不起作用:
List<String> names = ImmutableList.builder().add("Hello").add("World").build();
因为builder((方法是参数化的,java不会为了知道您可能想要什么而检查所有的add方法(也不会展望build((方法,然后返回变量声明(。因此,你可以看到这一点:
List<String> names = ImmutableList.<String>builder().add(...).build();