
在java泛型中,若需声明一个可接受所有实现特定接口(如isomething)的类的class引用,应使用通配符上界语法class<? extends isomething>,而非class<isomething>——后者表示“isomething接口自身的class对象”,而前者才表示“任何isomething子类型(包括实现类、子类)的class对象”。
要理解这一设计,关键在于明确 Class<T> 的泛型参数 T 的含义:它代表该 Class 对象所描述的运行时类的直接类型。例如:
- A.class 的类型是 Class<A>;
- B.class 的类型是 Class<B>;
- 而 ISomething.class 的类型是 Class<ISomething>(但注意:接口不能被实例化,ISomething.class 合法但无法用于 new ISomething())。
由于 A 和 B 都实现了 ISomething,它们在类型关系上满足 A <: ISomething 和 B <: ISomething(<: 表示“是子类型”),因此 Class<A> 和 Class<B> 都是 Class<? extends ISomething> 的子类型,可安全赋值:
interface ISomething { void doSomething(); }
class A implements ISomething {
public void doSomething() { System.out.println("A"); }
}
class B extends A {
@Override
public void doSomething() { System.out.println("B"); }
}
// ✅ 正确:声明支持所有 ISomething 实现类的 Class 引用
Class<? extends ISomething> clazz;
clazz = A.class; // OK: Class<A> → Class<? extends ISomething>
clazz = B.class; // OK: Class<B> → Class<? extends ISomething>
// clazz = String.class; // ❌ 编译错误:String 不实现 ISomething
// ⚠️ 注意:ISomething.class 也可赋值(因接口自身满足 ? extends ISomething),
// 但通常无实际构造意义,慎用于 newInstance()
clazz = ISomething.class; // 编译通过,但 clazz.asSubclass(ISomething.class) 会失败⚠️ 重要注意事项:
- Class<? extends ISomething> 允许你安全读取类型信息(如获取类名、注解、方法等),但不能直接调用 newInstance() 或 getConstructor(...).newInstance() 安全创建实例——因为编译器无法保证该类有无参构造器或是否为具体类(接口/抽象类无法实例化)。建议配合 asSubclass(ISomething.class) 做运行时校验:
try { ISomething instance = clazz.asSubclass(ISomething.class) .getDeclaredConstructor() .newInstance(); instance.doSomething(); } catch (Exception e) { throw new RuntimeException("Cannot instantiate " + clazz, e); } - 若仅需约束变量接收范围,无需实例化,Class<? extends ISomething> 是最简洁、类型安全的选择;
- 不要误用 Class<ISomething>(它只接受 ISomething.class)或 Class<?>(失去类型约束,不安全)。
综上,Class<? extends ISomething> 是表达“任意实现 ISomething 的具体类的 Class 对象”的标准、类型安全且符合 Java 泛型规范的方式。
立即学习“Java免费学习笔记(深入)”;










