解決」java.lang.UnsatisfiedLinkError: Native Library .dll already loaded in another classloader」的問題

JAVA啓動後,是通過JVM各級ClassLoader來加載各個類到內存。爲了更加了解加載過程,我經過分析和寫了一個簡單的ClassLoader來粗淺的分析它的原理。html

JVM的ClassLoader分三層,分別爲Bootstrap ClassLoader,Extension ClassLoader,System ClassLoader,他們不是類繼承的父子關係,是邏輯上的上下級關係。java

Bootstrap ClassLoader是啓動類加載器,它是用C++編寫的,從%jre%/lib目錄中加載類,或者運行時用-Xbootclasspath指定目錄來加載。web

Extension ClassLoader是擴展類加載器,從%jre%/lib/ext目錄加載類,或者運行時用-Djava.ext.dirs制定目錄來加載。api

System ClassLoader,系統類加載器,它會從系統環境變量配置的classpath來查找路徑,環境變量裏的.表示當前目錄,是經過運行時-classpath或-Djava.class.path指定的目錄來加載類。瀏覽器

能夠經過下面三條語句,輸入如今加載的各個classloader的加載路徑:app

        System.out.println("sun.boot.class.path:" + System.getProperty("sun.boot.class.path"));   jvm

        System.out.println("java.ext.dirs:" + System.getProperty("java.ext.dirs"));   jsp

        System.out.println("java.class.path:" +System.getProperty("java.class.path"));ui

 1 sun.boot.class.path:
 2     C:\D\programsoft\Java\jre1.8.0_45\lib\resources.jar;
 3     C:\D\programsoft\Java\jre1.8.0_45\lib\rt.jar;
 4     C:\D\programsoft\Java\jre1.8.0_45\lib\sunrsasign.jar;
 5     C:\D\programsoft\Java\jre1.8.0_45\lib\jsse.jar;
 6     C:\D\programsoft\Java\jre1.8.0_45\lib\jce.jar;
 7     C:\D\programsoft\Java\jre1.8.0_45\lib\charsets.jar;
 8     C:\D\programsoft\Java\jre1.8.0_45\lib\jfr.jar;
 9     C:\D\programsoft\Java\jre1.8.0_45\classes
10 
11 java.ext.dirs:
12     C:\D\programsoft\Java\jre1.8.0_45\lib\ext;
13     C:\WINDOWS\Sun\Java\lib\ext
14 
15 java.class.path:
16     C:\Users\Mat Lei\workspace\FlexAndJava_Server\build\classes;
17     C:\D\basic\programsoft\Tomcat 8.0\lib\annotations-api.jar;
18     C:\D\basic\programsoft\Tomcat 8.0\lib\catalina-ant.jar;
19     C:\D\basic\programsoft\Tomcat 8.0\lib\catalina-ha.jar;
20     C:\D\basic\programsoft\Tomcat 8.0\lib\catalina-storeconfig.jar;
21     C:\D\basic\programsoft\Tomcat 8.0\lib\catalina-tribes.jar;
22     C:\D\basic\programsoft\Tomcat 8.0\lib\catalina.jar;
23     C:\D\basic\programsoft\Tomcat 8.0\lib\ecj-4.4.2.jar;
24     C:\D\basic\programsoft\Tomcat 8.0\lib\el-api.jar;
25     C:\D\basic\programsoft\Tomcat 8.0\lib\jasper-el.jar;
26     C:\D\basic\programsoft\Tomcat 8.0\lib\jasper.jar;
27     C:\D\basic\programsoft\Tomcat 8.0\lib\jsp-api.jar;
28     C:\D\basic\programsoft\Tomcat 8.0\lib\servlet-api.jar;

 首先訪問項目A的一個界面,界面中調用了appletA,接着並無關閉瀏覽器而直接訪問項目B的界面,在界面中調用了appletB。appletA和appletB其實是同一個applet,只不過這個applet使用在了兩個項目中,而且兩個項目均是直接進行訪問。這時候在訪問appletB的時候,就會出現一個錯誤:google

xxx NOT loaded java.lang.UnsatisfiedLinkError : Native Library XXX.dll already loaded in another classloader

    若是訪問從appletB到appletA,那麼在訪問appletA時也會出現一樣的錯誤。
    由於,在一個標籤頁中,多個applet運行其實是運行在同一個jvm上,只是加載applet時使用了不一樣的classLoader。所以,無論是 appletA先運行仍是appletB先運行,最終狀況都是所依賴的dll都會被同一個jvm所加載,就會出現以上的錯誤了。

    在進行google以後,發現不少開發人員都碰到了一樣的問題,有的是由於在同一個javaEE容器如(weblogic,jboss)中部署了兩個都要 訪問同一個jni調用的項目,有的則是像筆者一樣的經歷。最後的結論便是,在一個jvm當中,是不容許加載一個dll兩次的。所以,後面的jni調用時, 嘗試再次加載同一個dll,這時候即會報上面的錯誤了。由於該錯誤,相對應的java類確定不能被初始化,所以相應的項目或者applet確定啓動不了了。

  解決辦法是把調用dll的jar複製到C:\D\programsoft\Java\jre1.8.0_45\lib下,即Bootstrap ClassLoader加載的目錄下,並在JRE System Library下添加該jar,並在

 

JRE System Library——Properties——Installed JREs——Edit——Add External JARs中選擇該jar。便可在Bootstrap ClassLoader加載該JAR,而且不受線程限制。

 

 

參考內容:

http://www.cnblogs.com/newstar/archive/2012/03/14/2396176.html

http://www.cnblogs.com/Lawson/archive/2012/07/31/2616623.html

相關文章
相關標籤/搜索