反射调用 Method.invoke() 比直接调用慢 10–100 倍,主因是 JVM 无法内联、运行时类型检查、每次权限校验及参数数组包装;setAccessible(true) 可提速 20–40% 但破坏封装且受模块系统限制;缓存 Method/Field 对象最有效;替代方案包括接口工厂、MethodHandle、VarHandle 和字节码生成。
Method.invoke() 比直接调用慢多少?不是“慢一点”,而是可能慢 10–100 倍,尤其在高频调用场景下。根本原因在于:JVM 无法对反射调用做内联(inlining)、类型检查推迟到运行时、每次调用都要校验访问权限、还要包装参数为 Object[] 数组。
obj.doSomething(123):编译期绑定,JIT 后几乎无开销method.invoke(obj, 123):每次都要查方法表、检查 Accessible、拆箱/装箱、异常封装(即使没异常也要准备 InvocationTargetException)setAccessible(true) 能提速但仍有隐患?它绕过 Java 访问控制检查(比如 private 字段读写),省掉一次安全校验,通常能提升 20–40% 的反射操作速度。但代价是:
SecurityManager 已废弃但仍有遗留环境)会直接拒绝该操作,抛出 SecurityException
setAccessible(true) 默认失败,需显式加 --add-opens 参数,否则抛 InaccessibleObjectException
Method / Field 对象有用吗?非常有用——这是最简单有效的优化手段。反射元对象(Method、Field、Constructor)本身是线程安全且可复用的,查找过程(clazz.getDeclaredMethod(...))才是重头开销。
MapmethodCache = new ConcurrentHashMap<>(); // 首次查找并缓存 Method m = clazz.getDeclaredMethod("getName"); m.setAccessible(true); methodCache.put("getName", m); // 后续直接用 m.invoke(obj); // 不再走 getDeclaredMethod
"getName:()Ljava/lang/String;"),否则重载方法会冲突invoke() 结果,那不是元数据,是执行行为有,但要看场景。纯性能优先时,优先考虑:
ServiceLoader 或 Spring Bean 查找,零反射开销ByteBuddy 或 CGLIB):在运行时生成代理类,首次生成稍慢,后续调用等同直接调用MethodHandle:比 Method.invoke() 快不少(约 2–3 倍),且支持更精细的权限控制,但 API 更底层、调试困难record)+ VarHandle(JDK 9+):对字段访问特别快,接近直接字段访问,但仅限 public 字段或配合 setAccessible
真正麻烦的是那些必须绕过编译期类型约束的场景——比如通用 ORM 的属性映射、JSON 反序列化。这时候
