How does one access a method from an external jar at runtime?

前端 未结 3 620
刺人心
刺人心 2021-01-12 15:51

This is a continuation of the question posted in: How to load a jar file at runtime

I am uncertain as to how to continue to the method invocation level. From my und

相关标签:
3条回答
  • 2021-01-12 16:09

    Sample Program:

    Project Printer:

    public class Printer {
    
        public void display(String printtext)
        {
            System.out.println(printtext);
        }
    
    }
    

    This project is exported as Printer.jar.

    Printer Class has method display() which takes string as input.

    Invoking code:

           URL url = new URL("file:Printer.jar"); 
           URLClassLoader loader = new URLClassLoader (new URL[] {url});
           Class<?> cl = Class.forName ("Printer", true, loader);
           String printString = "Print this";
           Method printit = cl.getMethod("display", String.class);
           Constructor<?> ctor = cl.getConstructor(); //One has to pass arguments if constructor takes input arguments.
           Object instance = ctor.newInstance();
           printit.invoke(instance, printString);
           loader.close ();
    

    Output: Print this

    0 讨论(0)
  • 2021-01-12 16:12

    The code example

    ClassLoader loader = URLClassLoader.newInstance(
        new URL[] { yourURL },
        getClass().getClassLoader()
    );
    Class<?> clazz = Class.forName("mypackage.MyClass", true, loader);
    Class<? extends Runnable> runClass = clazz.asSubclass(Runnable.class);
    // Avoid Class.newInstance, for it is evil.
    Constructor<? extends Runnable> ctor = runClass.getConstructor();
    Runnable doRun = ctor.newInstance();
    doRun.run();
    

    assumes that the class you are loading implements a particular interface Runnable, and therefore it's reasonale to cast to that type using asSubclass() and invoke run().

    What do you know about the classes you are loading? Can you assume that they implement a particualr interface? If so adjust the asSubClass() line to reference the interafce you prefer.

    Then, yes if you are working with instance methods create an instance using the contructor, ctor in the example.

    There is no starting of a thread in the example. Creating a new thread would just have needed a couple of lines more code

    Thread myThread = new Thread(doRun);
    myThread.start();
    
    0 讨论(0)
  • 2021-01-12 16:13

    Here is some reflection code that doesn't cast to an interface:

    public class ReflectionDemo {
    
      public void print(String str, int value) {
        System.out.println(str);
        System.out.println(value);
      }
    
      public static int getNumber() { return 42; }
    
      public static void main(String[] args) throws Exception {
        Class<?> clazz = ReflectionDemo.class;
        // static call
        Method getNumber = clazz.getMethod("getNumber");
        int i = (Integer) getNumber.invoke(null /* static */);
        // instance call
        Constructor<?> ctor = clazz.getConstructor();
        Object instance = ctor.newInstance();
        Method print = clazz.getMethod("print", String.class, Integer.TYPE);
        print.invoke(instance, "Hello, World!", i);
      }
    }
    

    Writing the reflected classes to an interface known by the consumer code (as in the example) is generally better because it allows you to avoid reflection and take advantage of the Java type system. Reflection should only be used when you have no choice.

    0 讨论(0)
提交回复
热议问题