关于泛型以及它们在这个问题背后真正做了什么有一个很好的讨论,所以我们都知道Vector<int[]>是一个整数数组的向量,HashTable<String, Person>是一个键是字符串和值Person的表。 然而,令我困惑的是Class<>的用法。

java类class也应该有一个模板名(或者是eclipse中的黄色下划线告诉我的)。我不知道我应该在里面写什么。Class对象的全部意义是当您没有关于对象的全部信息时,用于反射等。为什么它让我指定哪个类的类对象将持有?我显然不知道,否则我不会使用Class对象,我会使用特定的对象。


当前回答

我们所知道的是“任何类的所有实例都共享该类的相同java.lang.Class对象”

如)

Student a = new Student();
Student b = new Student();

a.getClass() == b.getClass()为真。

现在假设

Teacher t = new Teacher();

没有泛型是可能的。

Class studentClassRef = t.getClass();

但现在这是错误的…?

例如:public void printStudentClassInfo(类studentClassRef){}可以用Teacher.class调用

使用泛型可以避免这种情况。

Class<Student> studentClassRef = t.getClass(); //Compilation error.

那么T是什么?T为类型参数(也称为类型变量);用尖括号(<>)分隔,跟在类名后面。 T只是一个符号,就像写入类文件时声明的变量名(可以是任何名称)一样。稍后,在初始化过程中,T将被有效的类名所取代(HashMap<String> map = new HashMap<String>();)

例)类名<T1, T2,…, Tn >

所以Class<T>表示特定类类型'T'的类对象。

假设您的类方法必须使用如下所示的未知类型参数

/**
 * Generic version of the Car class.
 * @param <T> the type of the value
 */
public class Car<T> {
    // T stands for "Type"
    private T t;

    public void set(T t) { this.t = t; }
    public T get() { return t; }
}

这里T可以用作字符串类型的CarName

或T可以作为整数类型作为modelNumber,

OR T可以作为对象类型作为有效的汽车实例。

现在上面是一个简单的POJO,可以在运行时使用不同的方法。 例如,集合)List, Set, Hashmap是最好的例子,它们将根据T的声明与不同的对象一起工作,但一旦我们将T声明为字符串 例:HashMap<String> map = new HashMap<String>();然后它将只接受String Class实例对象。

泛型方法

泛型方法是引入自己的类型参数的方法。这类似于声明泛型类型,但类型参数的作用域仅限于声明它的方法。允许使用静态和非静态泛型方法,以及泛型类构造函数。

泛型方法的语法包括一个类型参数,位于尖括号内,出现在方法的返回类型之前。对于泛型方法,类型参数段必须出现在方法的返回类型之前。

 class Util {
    // Generic static method
    public static <K, V, Z, Y> boolean compare(Pair<K, V> p1, Pair<Z, Y> p2) {
        return p1.getKey().equals(p2.getKey()) &&
               p1.getValue().equals(p2.getValue());
    }
}

 class Pair<K, V> {

    private K key;
    private V value;
}

这里<K, V, Z, Y>是方法参数中使用的类型的声明,它应该在返回类型之前,这里是布尔类型。

在下面;类型声明<T>在方法级上是不需要的,因为它已经在类级上声明了。

class MyClass<T> {
   private  T myMethod(T a){
       return  a;
   }
}

但是下面是错误的,因为类级类型参数K、V、Z和Y不能在静态上下文中使用(这里是静态方法)。

class Util <K, V, Z, Y>{
    // Generic static method
    public static  boolean compare(Pair<K, V> p1, Pair<Z, Y> p2) {
        return p1.getKey().equals(p2.getKey()) &&
               p1.getValue().equals(p2.getValue());
    }
}

其他有效的场景包括

class MyClass<T> {

        //Type declaration <T> already done at class level
        private  T myMethod(T a){
            return  a;
        }

        //<T> is overriding the T declared at Class level;
        //So There is no ClassCastException though a is not the type of T declared at MyClass<T>. 
        private <T> T myMethod1(Object a){
                return (T) a;
        }

        //Runtime ClassCastException will be thrown if a is not the type T (MyClass<T>).  
        private T myMethod1(Object a){
                return (T) a;
        }       

        // No ClassCastException        
        // MyClass<String> obj= new MyClass<String>();
        // obj.myMethod2(Integer.valueOf("1"));
        // Since type T is redefined at this method level.
        private <T> T myMethod2(T a){
            return  a;
        }

        // No ClassCastException for the below
        // MyClass<String> o= new MyClass<String>();
        // o.myMethod3(Integer.valueOf("1").getClass())
        // Since <T> is undefined within this method; 
        // And MyClass<T> don't have impact here
        private <T> T myMethod3(Class a){
            return (T) a;
        }

        // ClassCastException for o.myMethod3(Integer.valueOf("1").getClass())
        // Should be o.myMethod3(String.valueOf("1").getClass())
    private  T myMethod3(Class a){
        return (T) a;
    }


        // Class<T> a :: a is Class object of type T
        //<T> is overriding of class level type declaration; 
        private <T> Class<T> myMethod4(Class<T> a){
            return  a;
        }
    }

最后,Static方法总是需要显式<T>声明;它不会从类级别class <T>派生。这是因为类级别T与实例绑定。

请阅读泛型的限制

通配符和子类型

泛型方法的类型参数

其他回答

只要使用牛肉类:

public <T> T beefmarshal( Class<beef> beefClass, InputBeef inputBeef )
     throws JAXBException {
     String packageName = docClass.getPackage().getBeef();
     JAXBContext beef = JAXBContext.newInstance( packageName );
     Unmarshaller u = beef.createBeef();
     JAXBElement<T> doc = (JAXBElement<T>)u.beefmarshal( inputBeef );
     return doc.getBeef();
}

再举一个例子,Class的泛型版本(Class<T>)允许编写如下所示的泛型函数。

public static <T extends Enum<T>>Optional<T> optionalFromString(
        @NotNull Class<T> clazz,
        String name
) {
    return Optional<T> opt = Optional.ofNullable(name)
            .map(String::trim)
            .filter(StringUtils::isNotBlank)
            .map(String::toUpperCase)
            .flatMap(n -> {
                try {
                    return Optional.of(Enum.valueOf(clazz, n));
                } catch (Exception e) {
                    return Optional.empty();
                }
            });
}

摘自Java文档:

[…] 更令人惊讶的是,class已经被泛化了。类字面量现在用作类型标记,提供运行时和编译时类型信息。这就启用了一种静态工厂的风格,例如新的AnnotatedElement接口中的getAnnotation方法:

<T extends Annotation> T getAnnotation(Class<T> annotationType); 

这是一个泛型方法。它从参数中推断出类型形参T的值,并返回一个适当的T实例,如下面的代码片段所示:

Author a = Othello.class.getAnnotation(Author.class);

在使用泛型之前,您必须将结果强制转换为Author。此外,您也没有办法让编译器检查实际参数是否表示Annotation的子类。[…]

我从来没用过这种东西。有人知道吗?

正如其他答案所指出的那样,有很多很好的理由来解释为什么这个类是通用的。然而,很多时候,您无法知道Class<T>要使用的泛型类型。在这些情况下,您可以简单地忽略黄色的eclipse警告,或者使用Class<?>……我就是这么做的;)

我们所知道的是“任何类的所有实例都共享该类的相同java.lang.Class对象”

如)

Student a = new Student();
Student b = new Student();

a.getClass() == b.getClass()为真。

现在假设

Teacher t = new Teacher();

没有泛型是可能的。

Class studentClassRef = t.getClass();

但现在这是错误的…?

例如:public void printStudentClassInfo(类studentClassRef){}可以用Teacher.class调用

使用泛型可以避免这种情况。

Class<Student> studentClassRef = t.getClass(); //Compilation error.

那么T是什么?T为类型参数(也称为类型变量);用尖括号(<>)分隔,跟在类名后面。 T只是一个符号,就像写入类文件时声明的变量名(可以是任何名称)一样。稍后,在初始化过程中,T将被有效的类名所取代(HashMap<String> map = new HashMap<String>();)

例)类名<T1, T2,…, Tn >

所以Class<T>表示特定类类型'T'的类对象。

假设您的类方法必须使用如下所示的未知类型参数

/**
 * Generic version of the Car class.
 * @param <T> the type of the value
 */
public class Car<T> {
    // T stands for "Type"
    private T t;

    public void set(T t) { this.t = t; }
    public T get() { return t; }
}

这里T可以用作字符串类型的CarName

或T可以作为整数类型作为modelNumber,

OR T可以作为对象类型作为有效的汽车实例。

现在上面是一个简单的POJO,可以在运行时使用不同的方法。 例如,集合)List, Set, Hashmap是最好的例子,它们将根据T的声明与不同的对象一起工作,但一旦我们将T声明为字符串 例:HashMap<String> map = new HashMap<String>();然后它将只接受String Class实例对象。

泛型方法

泛型方法是引入自己的类型参数的方法。这类似于声明泛型类型,但类型参数的作用域仅限于声明它的方法。允许使用静态和非静态泛型方法,以及泛型类构造函数。

泛型方法的语法包括一个类型参数,位于尖括号内,出现在方法的返回类型之前。对于泛型方法,类型参数段必须出现在方法的返回类型之前。

 class Util {
    // Generic static method
    public static <K, V, Z, Y> boolean compare(Pair<K, V> p1, Pair<Z, Y> p2) {
        return p1.getKey().equals(p2.getKey()) &&
               p1.getValue().equals(p2.getValue());
    }
}

 class Pair<K, V> {

    private K key;
    private V value;
}

这里<K, V, Z, Y>是方法参数中使用的类型的声明,它应该在返回类型之前,这里是布尔类型。

在下面;类型声明<T>在方法级上是不需要的,因为它已经在类级上声明了。

class MyClass<T> {
   private  T myMethod(T a){
       return  a;
   }
}

但是下面是错误的,因为类级类型参数K、V、Z和Y不能在静态上下文中使用(这里是静态方法)。

class Util <K, V, Z, Y>{
    // Generic static method
    public static  boolean compare(Pair<K, V> p1, Pair<Z, Y> p2) {
        return p1.getKey().equals(p2.getKey()) &&
               p1.getValue().equals(p2.getValue());
    }
}

其他有效的场景包括

class MyClass<T> {

        //Type declaration <T> already done at class level
        private  T myMethod(T a){
            return  a;
        }

        //<T> is overriding the T declared at Class level;
        //So There is no ClassCastException though a is not the type of T declared at MyClass<T>. 
        private <T> T myMethod1(Object a){
                return (T) a;
        }

        //Runtime ClassCastException will be thrown if a is not the type T (MyClass<T>).  
        private T myMethod1(Object a){
                return (T) a;
        }       

        // No ClassCastException        
        // MyClass<String> obj= new MyClass<String>();
        // obj.myMethod2(Integer.valueOf("1"));
        // Since type T is redefined at this method level.
        private <T> T myMethod2(T a){
            return  a;
        }

        // No ClassCastException for the below
        // MyClass<String> o= new MyClass<String>();
        // o.myMethod3(Integer.valueOf("1").getClass())
        // Since <T> is undefined within this method; 
        // And MyClass<T> don't have impact here
        private <T> T myMethod3(Class a){
            return (T) a;
        }

        // ClassCastException for o.myMethod3(Integer.valueOf("1").getClass())
        // Should be o.myMethod3(String.valueOf("1").getClass())
    private  T myMethod3(Class a){
        return (T) a;
    }


        // Class<T> a :: a is Class object of type T
        //<T> is overriding of class level type declaration; 
        private <T> Class<T> myMethod4(Class<T> a){
            return  a;
        }
    }

最后,Static方法总是需要显式<T>声明;它不会从类级别class <T>派生。这是因为类级别T与实例绑定。

请阅读泛型的限制

通配符和子类型

泛型方法的类型参数