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

來源:互聯網
上載者:User

標籤:

JAVA啟動後,是經過JVM各級ClassLoader來載入各個類到記憶體。為了更加瞭解載入過程,我通過分析和寫了一個簡單的ClassLoader來粗淺的分析它的原理。

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

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

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

System ClassLoader,系統類別載入器,它會從系統內容變數配置的classpath來尋找路徑,環境變數裡的.表示目前的目錄,是通過運行時-classpath或-Djava.class.path指定的目錄來載入類。

可以通過下面三條語句,輸入現在載入的各個classloader的載入路徑:

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

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

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

 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\classes10 11 java.ext.dirs:12     C:\D\programsoft\Java\jre1.8.0_45\lib\ext;13     C:\WINDOWS\Sun\Java\lib\ext14 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的時候,就會出現一個錯誤:

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

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

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.