java - 使用 Class 和/或 ClassLoader 实例化 Java 类

标签 java swing class jar classloader

我正在尝试根据 JFileChooser 返回给我的文件实例化一个类。到目前为止,我已经使 JFileChooser 正常工作,并且可以成功获取我正在查找的 .class 文件。但是,我在实际获取 .class 文件并用它创建对象时遇到了麻烦。

这是我正在编写的方法...

public static Agent loadAgentFromFile(File file){
        Agent agent = null;
        String fileName = file.getName();
        String filePath = file.getAbsolutePath();
        String parentDirectory = file.getParentFile().getName();

        String agentClassName = fileName.substring(0, fileName.lastIndexOf('.'));

        Object object = null;
        //System.out.println(filePath.contains(parentDirectory));
        //System.out.println(filePath.substring(0, filePath.indexOf(parentDirectory)));

        try {
            File f = new File(filePath.substring(0, filePath.indexOf(parentDirectory)));
            URL[] cp = {f.toURI().toURL()};
            URLClassLoader urlcl = new URLClassLoader(cp);

            Class agentDefinition = null;
            if (parentDirectory.equals("Agents"))
            {
                System.out.println(PACKAGE + agentClassName);
                agentDefinition = urlcl.loadClass(PACKAGE + agentClassName);
                //agentDefinition = Class.forName(PACKAGE + agentClassName);
            }
            else
            {
                System.out.println(PACKAGE + parentDirectory + "." + agentClassName);
                agentDefinition = urlcl.loadClass(parentDirectory + "." + agentClassName);
            }
            object = agentDefinition.newInstance();

            agent = (Agent)object;
        }
        catch (InstantiationException e) {

            JOptionPane.showMessageDialog(THIS,
                "The chosen class must not be an interface or be abstract, " +
                "and it must not have any arguments in its constructor.");
        }
        catch (IllegalAccessException e) {

            JOptionPane.showMessageDialog(THIS,
                "The chosen class's constructor must be public.");
        }
        catch (ClassNotFoundException e) {

            JOptionPane.showMessageDialog(THIS,
                "The chosen class cannot be found.  Make sure the .java file is compiled.");
        }
        catch (ClassCastException e) {

            JOptionPane.showMessageDialog(THIS,
                "The chosen class does not extend from the Agent class.");
        }
        catch (Exception e) {
            e.printStackTrace(); //prints the stack to see what exception occurred that we didn't catch

            JOptionPane.showMessageDialog(THIS,
                "The chosen class does not conform to the \"Agent\" class.");
        }

        return agent;
    }

但是,当调用此方法时,我实际上得到一个 NoClassDefFoundError,它被我的包罗万象的异常捕获:

Exception in thread "AWT-EventQueue-0" java.lang.NoClassDefFoundError: TestAgents/TestDFSAgent (wrong name: WumpusEnvironment/Model/Agent/TestAgents/TestDFSAgent)
    at java.lang.ClassLoader.defineClass1(Native Method)
    at java.lang.ClassLoader.defineClass(Unknown Source)
    at java.security.SecureClassLoader.defineClass(Unknown Source)
    at java.net.URLClassLoader.defineClass(Unknown Source)
    at java.net.URLClassLoader.access$100(Unknown Source)
    at java.net.URLClassLoader$1.run(Unknown Source)
    at java.net.URLClassLoader$1.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(Unknown Source)
    at java.lang.ClassLoader.loadClass(Unknown Source)
    at java.lang.ClassLoader.loadClass(Unknown Source)
    at WumpusEnvironment.Model.Agent.AgentLoader.loadAgentFromFile(AgentLoader.java:41)
    at WumpusEnvironment.View.MainWindow.ApplicationWindow.actionPerformed(ApplicationWindow.java:270)
    at javax.swing.AbstractButton.fireActionPerformed(Unknown Source)
    at javax.swing.AbstractButton$Handler.actionPerformed(Unknown Source)
    at javax.swing.DefaultButtonModel.fireActionPerformed(Unknown Source)
    at javax.swing.DefaultButtonModel.setPressed(Unknown Source)
    at javax.swing.AbstractButton.doClick(Unknown Source)
    at javax.swing.plaf.basic.BasicMenuItemUI.doClick(Unknown Source)
    at javax.swing.plaf.basic.BasicMenuItemUI$Handler.mouseReleased(Unknown Source)
    at java.awt.Component.processMouseEvent(Unknown Source)
    at javax.swing.JComponent.processMouseEvent(Unknown Source)
    at java.awt.Component.processEvent(Unknown Source)
    at java.awt.Container.processEvent(Unknown Source)
    at java.awt.Component.dispatchEventImpl(Unknown Source)
    at java.awt.Container.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.LightweightDispatcher.retargetMouseEvent(Unknown Source)
    at java.awt.LightweightDispatcher.processMouseEvent(Unknown Source)
    at java.awt.LightweightDispatcher.dispatchEvent(Unknown Source)
    at java.awt.Container.dispatchEventImpl(Unknown Source)
    at java.awt.Window.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
    at java.awt.EventQueue.access$200(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$1.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue.dispatchEvent(Unknown Source)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.run(Unknown Source)

我查看了以前的同类答案,并且我最初将 .class 文件放在整个包结构之外,因此我移动了它们。我的类文件现在与源文件位于同一文件夹中,但我仍然收到此错误。值得注意的是,我正在尝试在 Eclipse 中运行它,尽管将来整个项目将导出到 .jar 文件,该文件将实例化不在 Eclipse 中的类。包结构(因为此时所有内容都在 .jar 内)。

有人能给我指出正确的方向吗?我还希望您能指导我在导出到 .jar 文件时如何使您的建议继续发挥作用。我意识到 Eclipse 设置一些与 .java 文件相关的 .class 文件的方式有时会把这些事情弄乱。

谢谢!

最佳答案

Could someone point me in the right direction here?

我认为正确的方向是不要这样做。我建议您:

  1. 将插件代理类放入 JAR 文件中(JAR 中包含正确的路径)。

  2. 为每个 JAR 文件创建单独的类加载器或为所有 JAR 文件创建一个类加载器...具体取决于代理类是否需要彼此隔离。

您遇到的异常是因为类包名称与类加载器根目录中的“.class”文件的路径不匹配。在最坏的情况下,如果不移动“.class”文件,或者编写一个执行一些棘手操作的自定义类加载器,则无法使其匹配。即便如此,您仍然会遇到(潜在的)问题,即处理对其他“.class”文件的所需或不需要的依赖关系。

将插件放入 JAR 中可以避免这些问题。

关于java - 使用 Class 和/或 ClassLoader 实例化 Java 类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23024734/

相关文章:

Java - 使用 SQLite 数据库填充 JCombobox

java - 是否可以在服务器端加密并在客户端解密(使用 javascript)?

java - 二叉树中的节点交叉

java - 单击按钮时更新 jLabel

java - 如何找到我单击了哪个日历的按钮?

c++ - 当基类想要访问继承类的成员时

java - 错误 : Connecting MySQL database in Android app using PHP

c++ - 命名空间中的私有(private)类

C++:输入具有任何派生类 vector 的函数

java - 从应用程序邀请中获取推荐历史的选项?