使用代理时,接口在 ClassLoader 中不可见?

2022-09-03 14:20:46

当我尝试使用动态代理时,我看到以下异常

 com.intellij.rt.execution.application.AppMain DynamicProxy.DynamicProxy
Exception in thread "main" java.lang.IllegalArgumentException: interface Interfaces.IPerson is not visible from class loader
    at java.lang.reflect.Proxy.getProxyClass(Proxy.java:353)
    at java.lang.reflect.Proxy.newProxyInstance(Proxy.java:581)
    at DynamicProxy.Creator.getProxy(Creator.java:18)
    at DynamicProxy.DynamicProxy.main(DynamicProxy.java:54)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)

任何想法我需要做些什么来解决它


答案 1

如果这是 Web 应用程序,那么在创建动态代理时应使用 Web 应用程序类装入器。因此,例如,而不是:

Proxy.newProxyInstance(
  ClassLoader.getSystemClassLoader(),
  new Class < ? >[] {MyInterface.class},
  new InvocationHandler() {
    // (...)
});

尝试:

Proxy.newProxyInstance(
  this.getClass().getClassLoader(), // here is the trick
  new Class < ? >[] {MyInterface.class},
  new InvocationHandler() {
    // (...)
});

例如,tomcat 类装入器的层次结构(其他 Web 容器也有类似的)如下:

      Bootstrap
          |
       System
          |
       Common
       /     \
  Webapp1   Webapp2 ... 

它是 webapp 类加载器,它包含 Web 应用程序的 /WEB-INF/classes 目录中的类和资源,以及 Web 应用程序的 /WEB-INF/lib 目录下的 JAR 文件中的类和资源。


答案 2

当您尝试执行时,生成的实例与您创建代理时传递的实例不同。换句话说,你有两个同名的类对象,代理不确定哪一个是正确的(无论它们是否相同)。DynamicProxyClass.forName(youInterfaceClass.getName())java.lang.Class

通常,当您尝试代理的接口位于通过两个不同的类加载器(即Tomcat的“通用”和“应用程序”)加载的库中时,就会发生这种情况。

如果这没有帮助,请发布有关您的应用程序的更多信息 - 特别是如果您使用的是任何应用程序服务器,Spring或OSGi。


推荐