
在java泛型中,若需声明一个可接受所有实现特定接口(如isomething)的类的class引用,应使用通配符上界语法class extends isomething>,而非class
要理解这一设计,关键在于明确 Class
- A.class 的类型是 Class;
- B.class 的类型是 Class;
- 而 ISomething.class 的类型是 Class
(但注意:接口不能被实例化,ISomething.class 合法但无法用于 new ISomething())。
由于 A 和 B 都实现了 ISomething,它们在类型关系上满足 A <: isomething b class> 和 Class 都是 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 → Class extends ISomething>
clazz = B.class; // OK: Class → 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.class)或 Class>(失去类型约束,不安全)。
综上,Class extends ISomething> 是表达“任意实现 ISomething 的具体类的 Class 对象”的标准、类型安全且符合 Java 泛型规范的方式。
立即学习“Java免费学习笔记(深入)”;










