一个更优雅的解决方案,以避免在后台任务仍在运行时处理DbContext

时间:2013-08-09 15:06:34

标签: c# multithreading entity-framework

下面的代码按预期工作,但我不确定是否会使用最优雅的代码。

 using (DatabaseContext context = DatabaseContext.CreateContext(_incompleteConnString + prefix + campaignDBPlatform))
     Progress prog = new Progress();
     TaskFactory tf = new TaskFactory();
     var parent = tf.StartNew(() =>
      Parallel.ForEach(QuestionsLangConstants.questionLangs.Values, i =>
      {
        try
        {
         qrepo.UploadQuestions(QWorkBook.Worksheets[i.QSheet], i, prog);
        }
        catch (Exception ex)
        {
         MessageBox.Show(ex.Message);
        }
       })
    );
    prog.Show();
    var finalTask = parent.ContinueWith(i =>
    {
    using (DatabaseContext context2 = DatabaseContext.CreateContext(_incompleteConnString + prefix + campaignDBPlatform))
    {
      UploadedQuestionsRepliesRepository uqrepo = new    
      UploadedQuestionsRepliesRepository(context2);
      UploadedQuestionsReplies UQuestions = new UploadedQuestionsReplies() {
        Id = (int)uqrepo.getNextSeqValue(),
        FileName = "test", 
        RQType = Questions.QuestionsType.ToString(), 
        TimeStamp = DateTime.Now 
        };
     uqrepo.Insert(UQuestions);
     uqrepo.Save();
    }
  });
}

如果我没有添加context2 context在父母继续的时候被处理掉。如果我使用finalTask.Wait(),则UI会冻结。对我上面的内容有更好的解决方案吗?

1 个答案:

答案 0 :(得分:3)

你可以避免第一次使用。在这两种情况下使用上下文,如果您确定第二个任务将始终执行,请将其置于最后,否则添加一些额外的代码以确保始终处理您的上下文。

DatabaseContext context2 = DatabaseContext.CreateContext(_incompleteConnString + prefix + campaignDBPlatform)
//Initialize qrepo with the context here??
Progress prog = new Progress();
TaskFactory tf = new TaskFactory();
var parent = tf.StartNew(() =>
  Parallel.ForEach(QuestionsLangConstants.questionLangs.Values, i =>
  {
    try
    {
     qrepo.UploadQuestions(QWorkBook.Worksheets[i.QSheet], i, prog);
    }
    catch (Exception ex)
    {
     MessageBox.Show(ex.Message);
     // Do you really want to continue with the next task after the exception?
    }
   })
);
prog.Show();
var finalTask = parent.ContinueWith(i =>
{
  UploadedQuestionsRepliesRepository uqrepo = new UploadedQuestionsRepliesRepository(context2);
  UploadedQuestionsReplies UQuestions = new UploadedQuestionsReplies() {
        Id = (int)uqrepo.getNextSeqValue(), 
        FileName = "test", 
        RQType = Questions.QuestionsType.ToString(), 
        TimeStamp = DateTime.Now 
    };
    uqrepo.Insert(UQuestions);
    uqrepo.Save();
    context2.Dispose();
 });

}