Unable to load sqlite_jni: java.lang.UnsatisfiedLinkError: Native Library D:/resin/bin/sqlite_jni.dll already loaded in another

问题:web每次请求都会动态连接数据库;dbname是个数据库名参数,

private Connection getConn() throws SQLException{
        Connection conn=null;
        try {
            Class.forName("SQLite.JDBCDriver");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
        if(dbName!=null&&dbName.length()>4){
                conn=DriverManager.getConnection("jdbc:sqlite:/c://sqlite3//"+dbName+".db");
        }
        return conn;
    }

 

 

 

原因:重重加载同一个文件;

 

错误表明sqlite_jni.dll已经被JVM的ClassLoader load了。通过查阅资料发现Web


      Server的自动重启机制是产生这一问题的根源。当Resin重启包含sqlite_jni.dll的这个Web应用时,会因为Variant类的语句而自动执行jsqlite_jni.dll的加载。但重启Web应用并不是重启整个resin(即:上一次启动的JVM仍然存在),也就是说sqlite_jni.dll已经被加载过了,因此系统将抛出错误。而当我们手工重启resin时,则会将上一次启动的JVM关闭并重新启动,这时会正常加载sqlite_jni.dll。

 

 

解决:把sqlite.jar即sqlite jdbc驱动放到resion/tomcat中的公共lib目录下;而不是放在应用即:web-inf/lib下;

       因为sqlite.jar中,有sqlite_jni.dll文件加载的调用类;

 

 

 

 

问题解决灵感来自另一篇文章:http://blog.csdn.net/gavin_sw/archive/2007/04/10/1559733.aspx

 


 使用jacob时发生UnsatisfiedLinkError的问题 [原]

      问题描述:
         
      我在一个Web应用中(server是resin-ee-2.1.4)使用jacob完成word文档自动转换成pdf文档的功能(详见我另一篇日志),为了保证web应用的稳定性,我提供了resin每晚自动重启的机制,方法是利用定时servlet修改resin.conf文件。
            正在装载数据……
      但是当web应用重启后,jacob会报以下错误;
          Native Library C:/WINNT/system32/jacob.dll already loaded in another
      classloader
          java.lang.UnsatisfiedLinkError: Native Library
      C:/WINNT/system32/jacob.dll already loaded in another classloader
          at java.lang.ClassLoader.loadLibrary0(ClassLoader.java:1437)
          at java.lang.ClassLoader.loadLibrary(ClassLoader.java:1397)
          at java.lang.Runtime.loadLibrary0(Runtime.java:788)
          at java.lang.System.loadLibrary(System.java:832)
          at com.jacob.com.Variant.<clinit>(Variant.java)
          ……
          然后转pdf的功能会彻底失效,直到手工重启resin服务。

      问题分析:
          经过查看jacob源代码发现在Variant类中有一个静态方法:
          static {
            System.loadLibrary("jacob";
          }
          而UnsatisfiedLinkError错误表明jacob.dll已经被JVM的ClassLoader load了。通过查阅资料发现Web
      Server的自动重启机制是产生这一问题的根源。当Resin重启包含jacob的这个Web应用时,会因为Variant类的语句而自动执行jacob的加载。但重启Web应用并不是重启整个resin(即:上一次启动的JVM仍然存在),也就是说jacob已经被加载过了,因此系统将抛出UnsatisfiedLinkError错误。而当我们手工重启resin时,则会将上一次启动的JVM关闭并重新启动,这时会正常加载jacob。

      问题解决:
          Java API表明:JVM只允许一个默认的ClassLoader来load native
      library,同时并不提供专门的API来unload一个loaded native
      library,因此无法在我们的重启Web应用的代码中来手工清除已经load的jacob。为此我们必须保证在重启Web应用时不再重复加载jacob,具体方法是:将jacob.jar包放到Web
      Server的公共lib文件夹中(如:.../resin-ee-2.1.4/lib/),而不是Web应用的lib中(如:.../WEB-INF/lib/)。经过测试,自动重启Web应用后,问题不再出现。

      后记:
         虽然问题已解决,但总感觉有些不爽,毕竟没有从根源上解决。还希望看到此篇日志的网友继续研究,如果能找出unload loaded native
      library的方法,希望贴上来大家共享。

 

 

 

 

 

 

 

你可能感兴趣的:(jvm,Web,sqlite,ClassLoader,jni,library)