如何从运行时加载的类中调用构造函数? Java的

时间:2012-10-04 15:16:39

标签: java constructor classloader nosuchmethoderror dynamic-class-loaders

我有一个类加载器,它从/ plugins文件夹中的所有jar文件加载“main”类 这假设所有jar都有包含名为main的类的包插件。(插件名称)。每个主类都有一个名为main的构造函数。

类加载成功,但我需要知道如何从加载的类中调用主构造函数。

(此类/类在运行时加载)

我尝试过使用它:

Constructor c = cls.getConstructor(Integer.class); //line 41

Plugin plug = (Plugin) c.newInstance(0);

但是我收到了这个错误:

java.lang.NoSuchMethodException: plugin.myplugin.main.<init>(java.lang.Integer)  
at java.lang.Class.getConstructor0(Unknown Source)  
at java.lang.Class.getConstructor(Unknown Source)  
at hkr.classloader.PluginLoader.loadPlugins(PluginLoader.java:41)  
at hkr.core.startup.InitializeGame.inigame(InitializeGame.java:32)  
at hkr.launcher.main.LauncherMain.main(LauncherMain.java:16)  

package hackers.classloader;

import java.io.File;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;

import org.java.plugin.Plugin;

public class PluginLoader 
{
@SuppressWarnings({ "unused", "rawtypes", "resource" })
    public static void loadPlugins() throws NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException
    {
        Class cls = null;
        int x = hackers.core.startup.InitializeGame.map.size();
        for (int i = 1; i<=x;i++)
        {
            String className = hackers.core.startup.InitializeGame.map.get(i + "");

            File file  = new File(System.getProperty("user.dir") + File.separator + "plugins" + File.separator + className + ".jar");
            URL url = null;
            try {
                url = file.toURI().toURL();
            } catch (MalformedURLException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }  
            URL[] urls = new URL[]{url};
            ClassLoader cl = new URLClassLoader(urls);

            try {
                cls = cl.loadClass("plugin." + className + ".main");
            } catch (ClassNotFoundException e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }

            Constructor c = cls.getConstructor(Integer.TYPE);
            Plugin plug = (Plugin) c.newInstance(0);
        }
    }
}

1 个答案:

答案 0 :(得分:3)

如果您的构造函数采用java.lang.Integer,从我看到的,您的代码应该可以正常工作。

但是如果构造函数的唯一参数是int,则getConstructor将失败。在这种情况下,您必须使用Integer.TYPE而不是Integer.class。 我是对的,你需要做的是:

Constructor c = cls.getConstructor(Integer.TYPE);

编辑:根据您的修改和评论,有几个问题。

  1. 您要加载的类似乎没有任何显式构造函数,这意味着您只需要执行cls.getConstructor()
  2. 您要执行的内容(public static void main)是一种静态方法,通常您不需要类的实例。此外,由于用户@Eric B解释的原因,我不确定“main”是个好名字。
  3. 由于你想调用一个方法,你必须实例化构造函数并调用该方法。
  4. 根据我的理解,你想要执行的代码应该是这样的:

    Constructor c = cls.getConstructor(); // we get the implicit constructor without parameters
    Plugin plugin = (Plugin) c.newInstance(); // we instantiate it, no parameters
    
    Method m = cls.getDeclaredMethod("main", Integer.TYPE);
    m.invoke(plugin, 0); // we invoke the method "main" on our dynamically loaded class, with the 0 parameter.