贝利信息

Java反射机制与类加载的实际应用

日期:2026-01-14 00:00 / 作者:P粉602998670
反射访问私有字段需调用field.setAccessible(true)绕过安全检查,Class.forName()会初始化类而loadClass()不会,不同ClassLoader加载的同名类相互隔离,反射调用方法时JVM自动类型转换易掩盖参数错误。

反射获取私有字段值时为什么抛出 IllegalAccessException

Java 反射访问 private 字段默认被安全检查拦截,即使已用 getDeclaredField() 获取到字段对象。这不是权限配置问题,而是 JVM 运行时的默认行为。

Class.forName() 和 ClassLoader.loadClass() 的加载时机差异

两者都可按名称加载类,但触发类初始化的时机不同,直接影响静态块执行和常量赋值。

自定义 ClassLoader 加载相同全限定名类时的隔离性

不同 ClassLoader 实例加载的同名类,在 JVM 中视为完全不同的类型,不能互相赋值或强制转型,哪怕字节码一模一样。

public class PluginClassLoader extends ClassLoader {
    private final Map classBytes = new HashMap<>();

    public PluginClassLoader(ClassLoader parent) {
        super(parent);
    }

    @Override
    protected Class findClass(String name) throws ClassNotFoundException {
        byte[] bytes = classBytes.get(name);
        if (bytes == null) throw new ClassNotFoundException(name);
        // 关键:不调用 super.findClass(),避免双亲委派干扰
        return defineClass(name, bytes, 0, bytes.length);
    }
}

反射调用方法时参数类型不匹配却没报错?

使用 Method.invoke(obj, args...) 时,JVM 会尝试自动装箱/拆箱、基本类型与包装类互转、向上转型,导致“看似成功”但结果异常。

反射和类加载的边界很薄:一个 setAccessible(true)、一次错位的 forName、一个没处理好的类加载器隔离,都可能让逻辑在测试环境正常、上线后静默失败。这些不是边缘情况,而是实际支撑框架(如 Spring、MyBatis)运转的底层契约。