从同步控制器方法到异步控制器方法的正确转换(使用void时)

时间:2018-03-12 10:16:26

标签: c# asynchronous entity-framework-core asp.net-core-2.0

使用Entity Framework Core 2.0和.NET Core 2.0 Web API控制器我试图将它们从同步方法重写为异步方法。

这实际上对我的控制器方法只是查询数据很容易。

不幸的是,我将DbContext方法SaveChanges包装到一些辅助方法中,用于集中记录数据库更改。

在这里,我开始努力在异步上下文中正确使用void的组合。

代码说明: MyController是一个web api控制器。 PostMethod是一个(当前)同步post方法,接收请求模型,处理它,进行更改然后保存更改。

控制器中的SaveChangesWithLogs调用扩展方法并打印返回的日志。

扩展方法SaveChangesWithLogs生成日志条目(一些在保存之前,一些在保存之后),执行实际保存并返回日志。


  public class MyController : BaseController 
  {

    [HttpPost]
    public IActionResult PostMethod([FromBody]PostRequestModel request)
    {
        //do something
        SaveChangesWithLogs();
        //return created at
    }

    protected void SaveChangesWithLogs()
    {
      List logs = DbContext.SaveChangesWithLogs();
      foreach (string log in logs)
      {
        LogInfo(log); //just prints the generated logs
      }
    }

  }

  public static class MyExtensionMethod 
  {

    public static List SaveChangesWithLogs(this DbContext dbContext)
    {
      List logs = null;
      //pre-save prepare logs
      dbContext.SaveChanges();
      //post-save modifications of logs
      return logs;
    }
  }

在关于异步编程的教程中,他们提到方法应该一直异步。

所以我的问题: SaveChangesWithLogs方法的(方法签名)怎么样?


  public class MyController2 : BaseController 
  {

    [HttpPost]
    public async Task PostMethod([FromBody]PostRequestModel request)
    {
        //do something
        await SaveChangesWithLogs();
        //return created at
    }

    //does a correct implementation need async here too?
    protected void SaveChangesWithLogs()
    {
      List logs = await DbContext.SaveChangesWithLogs();
      foreach (string log in logs)
      {
        LogInfo(log); //just prints the generated logs
      }

      //return what???
    }

  }

  public static class MyExtensionMethod2 
  {

    public static async Task> SaveChangesWithLogs(this DbContext dbContext)
    {
      List logs = null;
      //pre-save prepare logs
      await dbContext.SaveChanges();
      //post-save modifications of logs
      return logs;
    }
  }

1 个答案:

答案 0 :(得分:1)

您只需像以前一样使用返回类型,并在其周围包装Task。没有比这更神奇了。

我猜测你的List类型属于某种类型,并且为了显示目的而添加了它。

protected async Task<bool> SaveChangesWithLogs()
    {
      List logs = await DbContext.SaveChangesWithLogs();
      foreach (string log in logs)
      {
        LogInfo(log); //just prints the generated logs
      }      
      return true;
    }

public static async Task<List<myLogType>> SaveChangesWithLogs(this DbContext dbContext)
{
    List<myLogType> logs = null;
    //pre-save prepare logs
    await dbContext.SaveChanges();
    //post-save modifications of logs
    return logs;
}