getclass() 返回对象实际运行时类型而非声明类型,因java动态绑定;它不能获取泛型信息,不适用于基本类型,且受类加载器隔离影响。

getClass() 不是反射的“入口”,而是对象自带的运行时类型快照——它返回的是当前实例确切所属的 Class 对象,不是父类、接口或泛型擦除后的类型。
为什么 getClass() 返回的不是声明类型?
Java 是动态绑定,变量声明类型(如 Object obj = new ArrayList())和实际运行时类型(ArrayList.class)可能不同。getClass() 只认堆里那个真实对象的类,不管左边怎么写。
- 常见错误现象:
Object obj = new String("a"); System.out.println(obj.getClass() == Object.class); // false - 使用场景:做类型安全的 clone、序列化前校验、日志中打印真实类型
- 注意:
getClass()不能用于基本类型(int、boolean等),它们没有类对象;要拿包装类的TYPE字段,比如int.class
getClass() 和 Class.forName() / MyClass.class 的区别
三者拿到的 Class 对象可能相同,但触发时机、加载行为、空安全性完全不同。
-
obj.getClass():必须有非 null 实例,运行时调用,100% 安全(null 调用直接抛NullPointerException) -
MyClass.class:编译期就确定,不触发类初始化,适合常量引用、注解处理器 -
Class.forName("pkg.MyClass"):通过字符串加载,会触发静态块执行,且可能抛ClassNotFoundException;类名错、包路径错、类没在 classpath 都会崩 - 性能影响:
getClass()几乎无开销;forName()涉及类加载器查找,慢且有异常风险
泛型擦除下 getClass() 能否获取泛型信息?
不能。运行时所有泛型都被擦除,new ArrayList<string>().getClass()</string> 和 new ArrayList<integer>().getClass()</integer> 返回的是同一个 Class 对象:ArrayList.class。
立即学习“Java免费学习笔记(深入)”;
- 常见错误现象:想靠
getClass()区分List<string></string>和List<integer></integer>,结果发现完全一样 - 真正能保留泛型信息的只有:字段/方法/父类上的
Type(需用getGenericXxx()系列方法)、带类型参数的ParameterizedType - 替代方案:如果必须运行时识别泛型用途,得靠额外传入
Class<t></t>参数(如Gson.fromJson(json, new TypeToken<list>>(){}.getType())</list>)
最易被忽略的一点:getClass() 返回的 Class 对象是 JVM 内部缓存的单例,但它的 isAssignableFrom()、isInstance() 行为受类加载器隔离影响——两个同名类由不同类加载器加载,哪怕字节码一模一样,getClass() 返回的 Class 对象也不等,isAssignableFrom 也会返回 false。










