jacob.dll already loaded in another classloader

问题描述:
我在一个Web应用中(tomcat)使用jacob完成word文档自动转换成html,但是一天后就上传不成文件了.查看tomcat的日志文件发现有如下异常:
严重: Servlet.service() for servlet AlanXUploadServlet threw exception
java.lang.UnsatisfiedLinkError: Native Library C:\Program Files\Java\jdk1.6.0_18\bin\jacob-1.15-M3-x86.dll already loaded in another classloader...
……
出现这样的问题,每当我手动重启tomcat服务后就有没问题了.

问题分析:
经过查看jacob源代码发现在Variant类中有一个静态方法:
static {
System.loadLibrary("jacob";
}
而UnsatisfiedLinkError错误表明jacob.dll已经被JVM的ClassLoader load了。通过查阅资料发现 Web Server的自动重启机制是产生这一问题的根源。当tomcat重启包含jacob的这个Web应用时,会因为Variant类的语句而自动执行 jacob的加载。但重启Web应用并不是重启整个tomcat(即:上一次启动的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文件夹中(如\Tomcat 6.0\lib),而不是Web应用的lib中(如:...\WEB-INF\lib\)。经过测试,自动重启Web应用后,问题不再出现。
[color=red]原因为:tomcat 启动后先将tomcat/lib目录下的jar包全部读入内存,如果webapps目录里的应用程序中WEB-INF/lib目录下有相同的包,将无法加载[/color]
后记:
虽然问题已解决,但总感觉有些不爽,毕竟没有从根源上解决。还希望看到此篇日志的网友继续研究,如果能找出unload loaded native library的方法,希望贴上来大家共享。

参考:
http://blog.csdn.net/gavin_sw/archive/2007/04/10/1559733.aspx
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值