系统中刚集成了一段Lucene的代码,用来实现简单的建立索引和搜索的功能。可是重新部署程序时,都会出现下面的异常:
Invocation of init method failed; nested exception is org.apache.lucene.store.LockObtainFailedException: Lock obtain timed out: NativeFSLock@/home/developer1/tomcat7/bin/index/write.lock
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.doCreateBean(AbstractAutowireCapableBeanFactory.java:527) ~[spring-beans-3.0.5.RELEASE.jar:3.0.5.RELEASE]
at org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory.createBean(AbstractAutowireCapableBeanFactory.java:456) ~[spring-beans-3.0.5.RELEASE.jar:3.0.5.RELEASE]
at org.springframework.beans.factory.support.AbstractBeanFactory$1.getObject(AbstractBeanFactory.java:291) ~[spring-beans-3.0.5.RELEASE.jar:3.0.5.RELEASE]
导致整个程序中的Spring Bean初始化都失败了。
进到索引目录中,发现里面就是一个write.lock。而IndexWriter的构造函数在试图获取另外一个IndexWriter已经加锁的索引目录时就会抛出一个LockObtainFailedException。
最直接的想法就是在程序undeploy时释放锁。好在Spring提供了非常方便的方式来实现这个功能。bean需要实现DisposableBean的借口,并且实现其destroy方法:
public void destroy() throws Exception { if(IndexWriter.isLocked(indexDirectory)){ writer.close(); IndexWriter.unlock(indexDirectory); } }
程序undeploy以后,再去索引目录下write.lock已经不见了。程序也可以顺利重新发布了。