74

実行中に新しいコード (拡張) をロードできる Java システムを構築するように依頼されました。コードの実行中に jar ファイルを再ロードするにはどうすればよいですか? または、新しいjarファイルをロードするにはどうすればよいですか?

明らかに、一定の稼働時間が重要であるため、既存のクラスを再ロードする機能を追加したいと考えています (複雑になりすぎない場合)。

私が気をつけなければならないことは何ですか?(実行時のクラスのリロードに関する質問と、新しいクラスの追加に関する質問の 2 つの異なる質問と考えてください)。

4

5 に答える 5

81

既存のデータで既存のクラスをリロードすると、問題が発生する可能性があります。

新しいコードを新しいクラス ローダーに比較的簡単にロードできます。

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();

使用されなくなったクラスローダーは、ガベージコレクションできます (ThreadLocal、JDBC ドライバーjava.beansなどを使用する場合によくあるように、メモリリークがない限り)。

オブジェクトデータを保持したい場合は、シリアライゼーションなどの永続化メカニズム、または慣れ親しんだものをお勧めします。

もちろん、デバッグシステムはより手の込んだことを行うことができますが、よりハックで信頼性が低くなります。

クラスローダーに新しいクラスを追加することができます。たとえば、 を使用しURLClassLoader.addURLます。ただし、クラスのロードに失敗した場合 (たとえば、クラスを追加していないため)、そのクラス ローダー インスタンスにロードされることはありません。

于 2008-10-11T21:57:09.493 に答える
36

これは私のために働く:

File file  = new File("c:\\myjar.jar");

URL url = file.toURL();  
URL[] urls = new URL[]{url};

ClassLoader cl = new URLClassLoader(urls);
Class cls = cl.loadClass("com.mypackage.myclass");
于 2009-03-23T13:44:31.907 に答える
8

実行中に新しいコードをロードできる Java システムを構築するように依頼されました。

まさにこの状況のた​​めに作られたOSGiをシステムに使用したい(または、少なくともそれに取り掛かる) ことをお勧めします。

クラスローダをいじることは、主にクラスの可視性がどのように機能するかという理由で、非常にトリッキーなビジネスであり、後でデバッグが困難な問題に遭遇したくありません。たとえば、多くのライブラリで広く使用されているClass.forName()は、断片化されたクラスローダー空間ではうまく機能しません。

于 2009-01-11T10:49:43.800 に答える
4

少しグーグルで検索したところ、このコードが見つかりまし

File file = getJarFileToLoadFrom();   
String lcStr = getNameOfClassToLoad();   
URL jarfile = new URL("jar", "","file:" + file.getAbsolutePath()+"!/");    
URLClassLoader cl = URLClassLoader.newInstance(new URL[] {jarfile });   
Class loadedClass = cl.loadClass(lcStr);   

このアプローチに関する意見/コメント/回答を共有できますか?

于 2008-10-11T21:55:31.883 に答える
2

次に示すように、org.openide.util.LookupとClassLoaderを使用して、Jarプラグインを動的にロードします。

public LoadEngine() {
    Lookup ocrengineLookup;
    Collection<OCREngine> ocrengines;
    Template ocrengineTemplate;
    Result ocrengineResults;
    try {
        //ocrengineLookup = Lookup.getDefault(); this only load OCREngine in classpath of  application
        ocrengineLookup = Lookups.metaInfServices(getClassLoaderForExtraModule());//this load the OCREngine in the extra module as well
        ocrengineTemplate = new Template(OCREngine.class);
        ocrengineResults = ocrengineLookup.lookup(ocrengineTemplate); 
        ocrengines = ocrengineResults.allInstances();//all OCREngines must implement the defined interface in OCREngine. Reference to guideline of implement org.openide.util.Lookup for more information

    } catch (Exception ex) {
    }
}

public ClassLoader getClassLoaderForExtraModule() throws IOException {

    List<URL> urls = new ArrayList<URL>(5);
    //foreach( filepath: external file *.JAR) with each external file *.JAR, do as follows
    File jar = new File(filepath);
    JarFile jf = new JarFile(jar);
    urls.add(jar.toURI().toURL());
    Manifest mf = jf.getManifest(); // If the jar has a class-path in it's manifest add it's entries
    if (mf
            != null) {
        String cp =
                mf.getMainAttributes().getValue("class-path");
        if (cp
                != null) {
            for (String cpe : cp.split("\\s+")) {
                File lib =
                        new File(jar.getParentFile(), cpe);
                urls.add(lib.toURI().toURL());
            }
        }
    }
    ClassLoader cl = ClassLoader.getSystemClassLoader();
    if (urls.size() > 0) {
        cl = new URLClassLoader(urls.toArray(new URL[urls.size()]), ClassLoader.getSystemClassLoader());
    }
    return cl;
}
于 2012-05-23T07:09:38.743 に答える