Max*_*sky 10 c# lucene asp.net-mvc lucene.net locking
我正在构建一个ASP.NET MVC站点,我想使用Lucene.Net进行搜索.我已经构建了一个SearchController及其所有方法,但是我在运行时遇到一个错误,这个错误是在首次初始化SearchController时发生的.
在SearchController中,这是我如何创建IndexWriter:
public static string IndexLocation = HostingEnvironment.MapPath("~/lucene");
public static Lucene.Net.Analysis.Standard.StandardAnalyzer analyzer = new Lucene.Net.Analysis.Standard.StandardAnalyzer();
public static IndexWriter writer = new IndexWriter(IndexLocation,analyzer);
Run Code Online (Sandbox Code Playgroud)
错误发生在最后一行.这是我得到的信息:
Lucene.Net.Store.LockObtainFailedException: Lock获取超时:SimpleFSLock @ C:\ Users\Username\Desktop\SiteSolution\Site\lucene\write.lock
此外,这是堆栈跟踪:
[LockObtainFailedException: Lock obtain timed out: SimpleFSLock@C:\Users\Username\Desktop\SiteSolution\Site\lucene\write.lock]
Lucene.Net.Store.Lock.Obtain(Int64 lockWaitTimeout) in C:\Users\Username\Desktop\Lucene.Net_2_9_2\src\Lucene.Net\Store\Lock.cs:107
Lucene.Net.Index.IndexWriter.Init(Directory d, Analyzer a, Boolean create, Boolean closeDir, IndexDeletionPolicy deletionPolicy, Boolean autoCommit, Int32 maxFieldLength, IndexingChain indexingChain, IndexCommit commit) in C:\Users\Username\Desktop\Lucene.Net_2_9_2\src\Lucene.Net\Index\IndexWriter.cs:1827
Lucene.Net.Index.IndexWriter.Init(Directory d, Analyzer a, Boolean closeDir, IndexDeletionPolicy deletionPolicy, Boolean autoCommit, Int32 maxFieldLength, IndexingChain indexingChain, IndexCommit commit) in C:\Users\Username\Desktop\Lucene.Net_2_9_2\src\Lucene.Net\Index\IndexWriter.cs:1801
Lucene.Net.Index.IndexWriter..ctor(String path, Analyzer a) in C:\Users\Username\Desktop\Lucene.Net_2_9_2\src\Lucene.Net\Index\IndexWriter.cs:1350
Site.Controllers.SearchController..cctor() in C:\Users\Username\Desktop\SiteSolution\Site\Controllers\SearchController.cs:95
[TypeInitializationException: The type initializer for 'Site.Controllers.SearchController' threw an exception.]
[TargetInvocationException: Exception has been thrown by the target of an invocation.]
System.RuntimeTypeHandle.CreateInstance(RuntimeType type, Boolean publicOnly, Boolean noCheck, Boolean& canBeCached, RuntimeMethodHandle& ctor, Boolean& bNeedSecurityCheck) +0
System.RuntimeType.CreateInstanceSlow(Boolean publicOnly, Boolean fillCache) +86
System.RuntimeType.CreateInstanceImpl(Boolean publicOnly, Boolean skipVisibilityChecks, Boolean fillCache) +230
System.Activator.CreateInstance(Type type, Boolean nonPublic) +67
System.Web.Mvc.DefaultControllerFactory.GetControllerInstance(RequestContext requestContext, Type controllerType) +80
[InvalidOperationException: An error occurred when trying to create a controller of type 'Site.Controllers.SearchController'. Make sure that the controller has a parameterless public constructor.]
System.Web.Mvc.DefaultControllerFactory.GetControllerInstance(RequestContext requestContext, Type controllerType) +190
System.Web.Mvc.DefaultControllerFactory.CreateController(RequestContext requestContext, String controllerName) +68
System.Web.Mvc.MvcHandler.ProcessRequestInit(HttpContextBase httpContext, IController& controller, IControllerFactory& factory) +118
System.Web.Mvc.MvcHandler.BeginProcessRequest(HttpContextBase httpContext, AsyncCallback callback, Object state) +46
System.Web.Mvc.MvcHandler.BeginProcessRequest(HttpContext httpContext, AsyncCallback callback, Object state) +63
System.Web.Mvc.MvcHandler.System.Web.IHttpAsyncHandler.BeginProcessRequest(HttpContext context, AsyncCallback cb, Object extraData) +13
System.Web.CallHandlerExecutionStep.System.Web.HttpApplication.IExecutionStep.Execute() +8682818
System.Web.HttpApplication.ExecuteStep(IExecutionStep step, Boolean& completedSynchronously) +155
Run Code Online (Sandbox Code Playgroud)
我该如何解决这个问题?
更新:我已经开始再次研究这个特定的项目了,似乎我还没有完全解决这个问题.
真正的问题是,该write.lock
文件不被索引的使用结束后删除.基于我接受的答案,我理解基本的实现逻辑,但我不确定我是否已正确实现它.以下是我班级中可能无效的其他一些方法:
public ActionResult Search(string query)
{
var reader = writer.GetReader(); // Get reader from writer
var searcher = new IndexSearcher(reader); // Build IndexSearch
//Execute search...
// Dispose of objects
searcher = null;
reader = null;
return View();
}
public void AddToIndex(Document doc)
{
writer.AddDocument(doc);
writer.Flush();
writer.Optimize();
writer.Flush();
}
private bool disposed = false;
protected override void Dispose(bool disposing)
{
if (!disposed)
{
if (disposing)
{
// Release managed resources.
}
try
{
writer.Close();
writer = null;
}
catch
{
}
// Release unmanaged resources.
// Set large fields to null.
// Call Dispose on your base class.
disposed = true;
}
base.Dispose(disposing);
}
Run Code Online (Sandbox Code Playgroud)
有什么想法吗?
Xod*_*rap 13
发生这种情况的原因是编写器创建了一个名为write.lock的空文件作为跨进程锁.当该文件存在时,Lucene假定有人在该目录上有写锁定.
如果错误地终止进程,则不会删除该文件.所以Lucene认为有人仍然坚持锁定.这就是为什么你应该总是有一个finally
关闭索引的声明.
如果你确定该文件存在错误(即没有运行Lucene进程),只需删除该文件即可.但是,您的索引可能处于损坏状态,因为写作显然是在中途终止.