HttpContext
当我的 ASP.NET MVC 应用程序(如 SignalR 集线器)中存在持久 HTTP 连接时,我在管理使用 StructureMap 的开放数据库连接的生命周期时遇到问题。
我的 DI 容器 StructureMap 将开放IDbConnection
注入到多个服务中。为确保这些数据库连接已关闭并正确处理,我呼吁ObjectFactory.ReleaseAndDisposeAllHttpScopedObjects()
该EndRequest
事件。
这对 MVC 控制器非常有用,直到需要数据库连接的服务被注入到 SignalR 集线器中,它为每个客户端保持持久的 HTTP 连接打开并最终使连接池饱和。
如果我将范围IDbConnection
限定为单例,则每个应用程序只打开一个连接并且池不会饱和,但如果连接被锁定或超时,这是一个坏主意。
那么也许有一种方法可以为我的 SignalR 集线器自定义数据库连接的范围?我尝试在每个 Hub 方法中解析一个服务实例,但这仍然会在 HttpContext 范围内实例化一个数据库连接,并在调用客户端的集线器连接期间保持打开状态。
当周围有持久的 HTTP 连接时,我应该如何在 HTTP 范围的上下文中使用 StructureMap 管理数据库连接的生命周期?
示例代码
典型服务
public class MyService
{
private IDbConnection _con;
public MyService(IDbConnection con)
{
_con = con;
}
public IEnumerable<string> GetStuff()
{
return _con.Select<string>("SELECT someString FROM SomeTable").ToList();
}
}
典型的 SignalR 集线器
public class MyHub : Hub
{
private MyService _service;
public MyHub(MyService service)
{
_service = service; // Oh Noes! This will open a database connection
// for each Client because of HttpContext scope
}
public Task AddMessage()
{
var result = _service.GetStuff();
// ...
}
}
结构图配置
For<IDbConnection>()
.HybridHttpOrThreadLocalScoped()
.Use(() => BaseController.GetOpenConnection(MyConnectionString));
全球.asax.cs
public class GlobalApplication : System.Web.HttpApplication
{
public GlobalApplication()
{
EndRequest += delegate
{
ObjectFactory.ReleaseAndDisposeAllHttpScopedObjects();
};
}
// ...
}