ASP.NET和多线程中每个请求的实体框架上下文

时间:2015-03-23 09:53:34

标签: c# asp.net multithreading entity-framework httpcontext

我的应用程序在ASP.NET MVC 4中。

我正在按照请求使用BDContext,正如这里的许多问题所推荐的那样。

我有:

public static class ContextPerRequest {
    private const string myDbPerRequestContext = "dbGeTraining_";

    public static DbGesForma_v2 db {
        get {
            if (!HttpContext.Current.Items.Contains(myDbPerRequestContext + HttpContext.Current.GetHashCode().ToString("x") + Thread.CurrentContext.ContextID.ToString())) {
                HttpContext.Current.Items.Add(myDbPerRequestContext + HttpContext.Current.GetHashCode().ToString("x") + Thread.CurrentContext.ContextID.ToString(), new DbGesForma_v2());
            }

            return HttpContext.Current.Items[myDbPerRequestContext + HttpContext.Current.GetHashCode().ToString("x") + Thread.CurrentContext.ContextID.ToString()] as DbGesForma_v2;
        }
    }

    /// <summary>
    /// Called automatically on Application_EndRequest()
    /// </summary>
    public static void DisposeDbContextPerRequest() {
        // Getting dbContext directly to avoid creating it in case it was not already created.
        var entityContext = HttpContext.Current.Items[myDbPerRequestContext + HttpContext.Current.GetHashCode().ToString("x") + Thread.CurrentContext.ContextID.ToString()] as DbGesForma_v2;
        if (entityContext != null) {
            entityContext.Dispose();
            HttpContext.Current.Items.Remove(myDbPerRequestContext + HttpContext.Current.GetHashCode().ToString("x") + Thread.CurrentContext.ContextID.ToString());
        }
    }
}

我将它处理在Application_EndRequest()方法中。这种方法长期以来运作良好。

现在我正在尝试使用这样的异步任务来实现:

 Task.Factory.StartNew(() => {
                DoSomething();
            });

这带来了一些问题。

  1. HttpContext在子线程中为null,并在上下文的键中使用。
  2. 即使我能够传递httpcontext或null检查它,如果子线程运行的时间比请求本身长,它将在线程完成之前处理,这将是有问题的。
  3. 任何解决方案?

2 个答案:

答案 0 :(得分:2)

我不确定您使用的是哪个版本的ASP.NET。无论如何,ASP.NET MVC(也是WebAPI)都有DependencyResolver,它支持那些'per-request'实例。

http://www.asp.net/mvc/overview/older-versions/hands-on-labs/aspnet-mvc-4-dependency-injection

另外,我建议您将DI框架与DependencyResolver一起使用,而不是实现每个请求实例工厂(或类似的东西)。大多数着名的DI框架都支持与ASP.NET集成。

例如;

答案 1 :(得分:0)

我找到了一个不太干扰的解决方案。 (我知道,Gongdo Gong解决方案要好得多,但在正在进行的项目中需要进行大量更改)

当我调用异步任务时,我会通过HttpContext,最后我处理了Context。

像这样:

      System.Web.HttpContext htcont = System.Web.HttpContext.Current;
      Task.Factory.StartNew(() => {
                System.Web.HttpContext.Current = htcont;
                DoSomething();
                ContextPerRequest.DisposeDbContextPerRequest();
       });

这样,HttpContext可以在子线程中使用,并且上下文会在作业结束时处理。