NoSuchMethodException при попытке создать экземпляр динамически загруженного класса с помощью конструктора, который включает в себя пользовательский класс - PullRequest
1 голос
/ 04 апреля 2019

Я пытаюсь создать экземпляр динамически загруженного класса (с пользовательским загрузчиком классов) с помощью конструктора, который содержит как объект String, так и созданный мной пользовательский класс.

Я заново создал проблему за пределами моего проекта, при этом предполагается, что A.class уже находится в целевой папке.

Основной метод:

public class simplifiedMainMethod {

static boolean propagate = false;
static Class<?> projectActions = null;
static Object oInstance;

public static void main (String[] args)
{
    String projectName = "A";
    String baseUrl = "https://www.google.com/";
    simplifiedReport reporter = new simplifiedReport();

    try 
    {
        ClassLoader cl = Thread.currentThread().getContextClassLoader();
        Reloader r = new Reloader(cl);

        if (propagate) 
            Thread.currentThread().setContextClassLoader(r);

        projectActions = r.loadClass("autobots_framework."+projectName);
        Constructor<?> c = projectActions.getConstructor(String.class, simplifiedReport.class);

        oInstance = c.newInstance(baseUrl, reporter);
    }
    catch (Exception e) {
        e.printStackTrace();
    }
}
}

simplifiedReport.java:

public class simplifiedReport 
{
   public simplifiedReport()
   {

   }
   public void testMethod()
   {
            System.out.println("Hello World");
   }
}

A.java:

public class A
{
        public A(String testString, simplifiedReport testReport) 
            { 
                   System.out.println(testString);
                   testReport.testMethod();
            }
}

Вот CustomClassLoader, который я использую:

public class ClassLoaderCompiler
{
    public ClassLoaderCompiler()
{}

public static boolean compileCode(String fileDirectory, String projectName) throws IOException
{   
    JavaCompiler compiler = new EclipseCompiler();
    DiagnosticCollector<JavaFileObject> diagnostics = new DiagnosticCollector<JavaFileObject>();
    StandardJavaFileManager fileManager = compiler.getStandardFileManager(diagnostics, null, null);
    Iterable<? extends JavaFileObject> compilationUnits = fileManager.getJavaFileObjectsFromStrings(Arrays.asList(fileDirectory));
    JavaCompiler.CompilationTask task = compiler.getTask(null, fileManager, diagnostics, null, null, compilationUnits);
    try
    {
        task.call();
        fileManager.close();
        return true;
    }
    catch(Exception e)
    {
        fileManager.close();
        return false;
    }
}

public static void moveClassFile(String projectName)
{
    try
    {
        Files.deleteIfExists(new File(System.getProperty("user.dir")+"/target/classes/autobots_framework/"+projectName+".class").toPath());
        Files.deleteIfExists(new File(System.getProperty("user.dir")+"/target/classes/autobots_framework/L_"+projectName+".class").toPath());

        if (Files.exists(Paths.get(System.getProperty("user.dir")+"/projects/"+projectName+"/"+projectName+".class")))
        {
            Path temp = Files.move (Paths.get(System.getProperty("user.dir")+"/projects/"+projectName+"/"+projectName+".class"),Paths.get(System.getProperty("user.dir")+"/target/classes/autobots_framework/"+projectName+".class")); 
            Path tempLocator = Files.move (Paths.get(System.getProperty("user.dir")+"/projects/"+projectName+"/L_"+projectName+".class"),Paths.get(System.getProperty("user.dir")+"/target/classes/autobots_framework/L_"+projectName+".class"));
        }
    }
    catch (Exception e)
    {
        e.printStackTrace();
    }
}

}

/**
 * adapted from http://stackoverflow.com/a/3971771/7849
 */
class Reloader extends ClassLoader {

static URL url;
ClassLoader orig;

Reloader(ClassLoader orig) {
    this.orig = orig;
}

@Override
public Class<?> loadClass(String s) {
    return findClass(s);
}

@Override
public Class<?> findClass(String s) {
    try {
        byte[] bytes = loadClassData(s);
        return defineClass(s, bytes, 0, bytes.length);
    } catch (IOException ioe) {
        try {
            return super.loadClass(s);
        } catch (ClassNotFoundException ignore) {
            ignore.printStackTrace(System.out);
        }
        ioe.printStackTrace(System.out);
        return null;
    }
}

private byte[] loadClassData(String className) throws IOException {
    try {
        /*
         * get the actual path using the original classloader 
         */
        Class<?> clazz = orig.loadClass(className);
        url = clazz.getResource(clazz.getSimpleName() + ".class");

        /*
         * force reload
         */
        File f = new File(url.toURI());
        int size = (int) f.length();
        byte buff[] = new byte[size];
        FileInputStream fis = new FileInputStream(f);
        DataInputStream dis = new DataInputStream(fis);
        dis.readFully(buff);
        dis.close();
        return buff;
    } catch (Exception ex) {
        throw new IOException(ex);
    }
}
}

Я получаю эту ошибку в результате:

java.lang.NoSuchMethodException: autobots_framework.A.<init>(java.lang.String, autobots_framework.simplifiedReport)
at java.lang.Class.getConstructor0(Unknown Source)
at java.lang.Class.getConstructor(Unknown Source)

Я подумал, что это из-за несовпадения параметров, поэтому для подтверждения я попытался найти то, что неявно ищет конструктор:

for (Constructor c : projectActions.getDeclaredConstructors())
{
      System.out.print(c.toGenericString());
}

Даем этот результат:

 public autobots_framework.A(java.lang.String,autobots_framework.simplifiedReport)

К сожалению, это очень похоже на то, что я здесь написал:

Constructor<?> c = projectActions.getConstructor(String.class, simplifiedReport.class);
oInstance = c.newInstance(baseUrl, reporter);

Я понимаю, что это как-то связано с моим классом, так как когда я опускаю simplifiedReport в конструкторе и только создаю его экземпляр с помощью String.class, я успешен. Какой модификатор мне не хватает в моем классе, чтобы заставить это работать?

Это все функция моего приложения, чтобы динамически компилировать код и затем создавать его экземпляры для запуска пользовательского кода и методов.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...