java - OSGi 客户端从 JavaEE 服务器访问 EJB 中的 RMI ClassCastException

标签 java jakarta-ee generics glassfish osgi

我有一个 OSGi 应用程序。使用 EJB context.lookup 时,我必须将 Thread 上下文类加载器设置为捆绑类加载器才能进行转换。像这样:

Thread.currentThread().setContextClassLoader(getClass().getClassLoader());
Entity entity=bean.getOne();
System.out.println(entity.getClass().getClassLoader());

输出为

org.apache.felix.framework.BundleWiringImpl@7468776f

这段代码有效。如果我有ArrayList则无法转换的问题

Thread.currentThread().setContextClassLoader(getClass().getClassLoader());
ArrayList<Entity> entities=bean.getMany();

此代码返回 ClassCastException。

检查

ArrayList<Entity> temp=new ArrayList<Entity>();
System.out.println(temp.getClass().getClassLoader());

返回 NULL - 它 means引导类。如何解决?

编辑:

最有趣的是,带有 String 的 ArrayList 可以工作,经典数组也可以工作,但是 ArrayList 和带有 Entiry 的 ArrayList 不起作用。

Class Bean {
....
  @Override //THIS DOESN'T WORK
    public ArrayList<Entity> readMany() {
        Entity dir1=new Entity();
        dir1.setContent("1 test");
        Entity dir2=new Entity();
        dir2.setContent("2 test");
        ArrayList<Entity> result=new ArrayList<>();
        result.add(dir1);result.add(dir2);
        return result;
    }

    @Override //THIS WORKS
    public ArrayList<String> readMany2() {
        String str1=new String("1 test");
        String str2=new String("2 test");
        ArrayList<String> result=new ArrayList<>();
        result.add(str1);
        result.add(str2);
        return result; 
    }

@Override //THIS WORKS
public Entity[] readArray() {
    ArrayList<Entity> al=readMany();
    Entity[] ar=new Entity[al.size()];
    for (int i = 0; i < al.size(); i++) {
        ar[i]=al.get(i);
    }
    return ar;
}

@Override //THIS DOESN'T WORK
public ArrayList readSimpleArrayList() {
    ArrayList<Entity> gal=readMany();
    ArrayList al= new ArrayList();
    for (Entity obj : gal) {
        al.add(obj);
    }
    return al;
}
...
}

这是日志

java.lang.ClassCastException: com.test.cmn.shd.base.dir.language.LanguageDirEntity cannot be cast to com.test.cmn.shd.base.dir.language.LanguageDirEntity at com.test.cmn.dt.base.Activator.start(Activator.java:83) at org.apache.felix.framework.util.SecureAction.startActivator(SecureAction.java:645) at org.apache.felix.framework.Felix.activateBundle(Felix.java:1977) at org.apache.felix.framework.Felix.startBundle(Felix.java:1895) at org.apache.felix.framework.BundleImpl.start(BundleImpl.java:944) at org.apache.felix.framework.BundleImpl.start(BundleImpl.java:931) at com.test.cmn.dt.loader.LoaderModel.startCoreModule(LoaderModel.java:149) at com.test.cmn.dt.loader.LoaderModel.access$100(LoaderModel.java:39) at com.test.cmn.dt.loader.LoaderModel$InstallAndStartModuleWorker.doInBackground(LoaderModel.java:79) at com.test.cmn.dt.loader.LoaderModel$InstallAndStartModuleWorker.doInBackground(LoaderModel.java:73) at javax.swing.SwingWorker$1.call(SwingWorker.java:296) at java.util.concurrent.FutureTask.run(FutureTask.java:262) at javax.swing.SwingWorker.run(SwingWorker.java:335) at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145) at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615) at java.lang.Thread.run(Thread.java:744)

编辑 2 - 完整代码。此代码在客户端计算机上执行。 JavaEE (GF4) 正在服务器上运行。 osgi 包分为三个:服务器版、客户端版和共享版。共享的副本位于服务器和客户端上,并包含 LanguageDirBeanRemote 和 LanguageDirEntity。

ClassLoader thatLoader = Thread.currentThread().getContextClassLoader();
Thread.currentThread().setContextClassLoader(getClass().getClassLoader());
try {
       Properties jndiProps = new Properties();
       jndiProps.put("java.naming.factory.initial", "com.sun.enterprise.naming.impl.SerialInitContextFactory");
       jndiProps.put("java.naming.factory.url.pkgs", "com.sun.enterprise.naming");
       jndiProps.put("java.naming.factory.state", "com.sun.corba.ee.impl.presentation.rmi.JNDIStateFactoryImpl");
       jndiProps.setProperty("org.omg.CORBA.ORBInitialHost", "x.x.x.x");
       jndiProps.setProperty("org.omg.CORBA.ORBInitialPort", "3700");
       InitialContext ctx = new InitialContext(jndiProps);
       LanguageDirBeanRemote bean=(LanguageDirBeanRemote)ctx.lookup("java:global/...");
       ArrayList<LanguageDirEntity> elements=bean.readDirectory();
      System.out.println("HERE I GET THE ERROR:"+elements.get(0).getContent());
} finally {
Thread.currentThread().setContextClassLoader(thatLoader);
}

编辑3 我开了一个bug

最佳答案

X 无法转换为 X”,在类加载器/OSGi 上下文中,通常意味着系统中存在两个类加载器,它们都已加载类X,并且您正在尝试将使用一个类加载器的类副本创建的实例传递给需要另一个类加载器的副本的代码。

修复要么是确保任何类加载器中只存在该类的一个实例,并且由所有代码共享(最简单),要么确保实例永远不会跨越类加载器上下文边界,或者(更脆弱)使确保每个人都找到该类的完全相同的实现(该类及其继承的所有内容的字节码完全相同)。

关于java - OSGi 客户端从 JavaEE 服务器访问 EJB 中的 RMI ClassCastException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23174582/

相关文章:

java - 如何让 Java 32 位网络应用程序在完全 64 位机器上运行?

java - 上下文参数值不是字符串

java - 在它的抽象父类(super class)中使用子类的泛型类型?

java - 在 java 中解析 HTTP JSONObject 响应

java - 如果括号匹配则返回 true 的 boolean 方法

java - 无法看到图像(Imageloader.java)

java - NetBeans 平台主机

java - GlassFish JDBC 领域组成员资格

java - 您如何访问泛型的 Class 对象?

C++ 类模板作为属性类型