10

在 Java 规范 ( http://docs.oracle.com/javase/specs/jls/se7/html/jls-15.html#jls-15.9 ) 中, new 具有以下形式:

ClassInstanceCreationExpression ::=
| new TypeArguments_opt TypeDeclSpecifier TypeArgumentsOrDiamond_opt
    ( ArgumentListopt ) ClassBodyopt
| Primary . new TypeArguments_opt Identifier TypeArgumentsOrDiamond_opt
    ( ArgumentListopt ) ClassBodyopt

new 之后的第一个可选类型参数列表的目的是什么?我在阅读第 15.9 节时无法找到它(所有对类型参数列表的引用似乎都引用了类型/标识符之后的列表)。在标准 Java 编译器上测试随机位会产生令人困惑的结果:

public class Foo<T> { }
// ...
Foo<Integer> t1 = new <Integer> Foo<Integer>();  // works
Foo<Integer> t2 = new <Integer> Foo();           // works -- unchecked warning missing the type arg after Foo
Foo<Integer> t3 = new <Boolean> Foo<Integer>();  // works
Foo<Integer> t4 = new <Float, Boolean> Foo<Integer>();  // works
Foo<Integer> t5 = new <NotDefined> Foo<Integer>();  // fails -- NotDefined is undefined

在这些简单的例子中,第一个参数列表似乎没有做任何有意义的事情,尽管它解析并检查其参数的有效性。

4

2 回答 2

13

构造函数也可以声明类型参数

public class Main {     
    public static class Foo<T> {
        public <E> Foo(T object, E object2) {

        }
    }
    public static void main(String[] args) throws Exception {
        Foo<Integer> foo = new <String> Foo<Integer>(1, "hello");           
    }    
}

这就是<String>前面的构造函数调用的目的。它是构造函数的类型参数。

以下

Foo<Integer> foo = new <String> Foo<Integer>(1, new Object());

失败了

Main.Foo 类型的参数化构造函数 Foo(Integer, String) 不适用于参数 (Integer, Object)

在你的最后

Foo<Integer> t5 = new <NotDefined> Foo<Integer>();  // fails -- NotDefined is undefined

NotDefined只是不是在编译期间找到的类型。如果是,它只会给你一个警告,它是unused

于 2013-09-08T21:47:43.343 回答
3

你可以为方法调用添加无用的类型参数,比如奇怪的东西

Math.<Runnable>max(1,2);

System.out.<Button>println();

请参阅http://docs.oracle.com/javase/specs/jls/se7/html/jls-15.html#jls-15.12.2.1-200-E

非泛型方法可能适用于提供显式类型参数的调用。事实上,它可能会被证明是适用的。在这种情况下,类型参数将被简单地忽略。

该规则源于兼容性问题和可替代性原则。由于接口或超类可以独立于它们的子类型进行泛型,我们可以用非泛型方法覆盖泛型方法。但是,覆盖(非泛型)方法必须适用于对泛型方法的调用,包括显式传递类型参数的调用。否则,子类型将无法替代其泛型的超类型。

于 2013-09-08T22:42:45.677 回答