是什么导致了“;由于一个被放弃的互斥体而完成了等待;
本文关键字:等待 是什么 于一个 放弃 | 更新日期: 2023-09-27 17:58:35
我有一个连接类来处理Informix数据库查询。它有两个功能;一个用于执行简单查询,另一个用于返回数据表。偶尔(尤其是当我让会话停留一段时间,比如十分钟),当请求新信息时,我会在conn.open命令上得到一个被放弃的互斥错误。
这是有问题的代码:
public DataTable CallDtQuery(string query)
{
DataTable dt = new DataTable();
using (IBM.Data.Informix.IfxConnection conn = new
IBM.Data.Informix.IfxConnection(sqlConnection))
{
try
{
IBM.Data.Informix.IfxDataAdapter adapter = new IfxDataAdapter();
adapter.SelectCommand = new IBM.Data.Informix.IfxCommand(query, conn);
conn.Open(); //Error location.
adapter.Fill(dt);
conn.Close();
}
catch (IBM.Data.Informix.IfxException ex)
{
LogError(ex, query);
SendErrorEmail(ex, query);
DisplayError();
}
}
return dt;
}
此外,这里还有一个简单的查询函数,它是应用程序中唯一连接到数据库的其他函数:
public string CallSimpleQuery(string query, string command)
{
string result = "";
using (IBM.Data.Informix.IfxConnection conn = new
IBM.Data.Informix.IfxConnection(sqlConnection))
{
try
{
IBM.Data.Informix.IfxDataAdapter adapter = new IfxDataAdapter();
conn.Open();
switch (command)
{
case "UPDATE":
adapter.UpdateCommand = new IBM.Data.Informix.IfxCommand(query, conn);
result = adapter.UpdateCommand.ExecuteNonQuery().ToString();
break;
case "DELETE":
adapter.DeleteCommand = new IBM.Data.Informix.IfxCommand(query, conn);
result = adapter.DeleteCommand.ExecuteNonQuery().ToString();
break;
case "SELECT":
adapter.SelectCommand = new IBM.Data.Informix.IfxCommand(query, conn);
result = adapter.SelectCommand.ExecuteScalar().ToString();
break;
case "INSERT":
adapter.InsertCommand = new IBM.Data.Informix.IfxCommand(query, conn);
result = adapter.InsertCommand.ExecuteNonQuery().ToString();
break;
}
conn.Close();
}
catch (IBM.Data.Informix.IfxException ex)
{
LogError(ex, query);
SendErrorEmail(ex, query);
DisplayError();
}
}
return result;
}
以下是生成的错误:
Error Message = The wait completed due to an abandoned mutex.
Message Source:
mscorlib
=============================
Message Target:
Boolean WaitOne(Int64, Boolean)
=============================
Stack Trace:
at System.Threading.WaitHandle.WaitOne(Int64 timeout, Boolean exitContext)
at System.Threading.WaitHandle.WaitOne(Int32 millisecondsTimeout, Boolean exitContext)
at System.Threading.WaitHandle.WaitOne()
at IBM.Data.Informix.IfxConnPoolManager.GetPool(IfxConnSettings key)
at IBM.Data.Informix.IfxConnPoolManager.Open(IfxConnection connection)
at IBM.Data.Informix.IfxConnection.Open()
at XXX.Connections.CallDtQuery(String query) in d:'Inetpub'wwwroot'intranet'CWSheet-test2'App_Code'Connections.cs:line 75
at XXX.details.Page_Load(Object sender, EventArgs e) in d:'Inetpub'wwwroot'intranet'CWSheet-test2'Details.aspx.cs:line 29
at System.Web.Util.CalliHelper.EventArgFunctionCaller(IntPtr fp, Object o, Object t, EventArgs e)
at System.Web.Util.CalliEventHandlerDelegateProxy.Callback(Object sender, EventArgs e)
at System.Web.UI.Control.OnLoad(EventArgs e)
at System.Web.UI.Control.LoadRecursive()
at System.Web.UI.Page.ProcessRequestMain(Boolean includeStagesBeforeAsyncPoint, Boolean includeStagesAfterAsyncPoint)
当错误发生时,它不是进入catch块并运行Log、Send和Display错误函数,而是进入global.asax上的Application_error块。由于所有内容都封装在try/catch块中,我不确定是什么原因导致了这种情况。此外,无论出于何种原因,应用程序有时都会挂在CallDtQuery上。我将在表单视图中翻阅记录页,它会突然挂在CallDtQuery请求上。有时它可能会在一两分钟后通过,有时它会无限期挂起,直到应用程序在30分钟后超时。
我读过一些关于互斥的文章,但以前从未使用过它。正在使用的任何互斥都是由ASP.NET应用程序自动生成的。考虑到这一点,我真的不知道如何解决这个问题。有什么建议吗?
所以问题出在我使用的IBM.Data.Informix.dll版本(2.90)上。我在这里找到了解释问题的文档:http://www.iiug.org/forums/development-tools/index.cgi/read/109
当我更新到一个新版本(3.50)时,被放弃的Mutex错误就消失了。间歇性挂起问题也消失了。
好吧,假设你没有做任何奇怪的线程处理,有几件事需要考虑
- informix C#类在并发管理方面存在一些问题(我现在还记得这个APAR)
- 我不知道为什么,但在打开连接之前创建命令感觉很奇怪
- 我会将ifxcommand处理到,特别是在IfxConnection上调用dispose会自动调用Close方法,也许在双关闭中,一些句柄会混淆,看看这个