entity.ToListAsync().Where 或 entity.Where()?

Kyt*_*yte 1 c# async-await asp.net-web-api entity-framework-6

我很难决定哪种方式最适合在我的 WebAPI 2 后端查询我的 SQL Server

我试图尽可能多地使用 async/await,但我发现当我返回整个集合时,没有可用的 async 选项。

哪种方式最好?

[ResponseType(typeof(List<ExposedPublisher>))]
[HttpGet]
public async Task<IHttpActionResult> GetPublisher()
{
    var list = new List<PublisherWithMedia>();
    foreach (var publisher in _db.Publisher.Where(e => e.IsDeleted == false))
    {
        var pub = new PublisherWithMedia()
            {
                Id = publisher.Id,
                Name = publisher.Name,
                Mediae = new List<WebClient.Models.Media>()
            };
            foreach (var media in publisher.Media)
            {
                pub.Mediae.Add(ApiUtils.GetMedia(media));
            }
        list.Add(pub);
    }
    return Ok(list);
}
Run Code Online (Sandbox Code Playgroud)

或者

[ResponseType(typeof(List<PublisherWithMedia>))]
[HttpGet]
public async Task<IHttpActionResult> GetPublisher()
{
    var list = new List<PublisherWithMedia>();
    var entity = await _db.Publisher.ToListAsync();
    foreach (var publisher in entity.Where(e => e.IsDeleted == false))
    {
        var pub = new PublisherWithMedia()
            {
                Id = publisher.Id,
                Name = publisher.Name,
                Mediae = new List<WebClient.Models.Media>()
            };
            foreach (var media in publisher.Media)
            {
                pub.Mediae.Add(ApiUtils.GetMedia(media));
            }
        list.Add(pub);
    }
    return Ok(list);
}
Run Code Online (Sandbox Code Playgroud)

该操作可能会导致非常大的结果集,因此直接在数据库上进行过滤是有意义的,特别是因为随着时间的推移,已删除记录的数量可能会超过未删除记录的数量。但是,由于结果较大且需要查询子项(媒体),因此异步执行操作也是有意义的,因为它应该非常耗时。可悲的是,Where()在这种情况下没有异步。

还是有我不知道的第三种方式?

SO *_*ood 5

您可以两全其美:在数据库中过滤并异步执行查询:

var publishers = await _db.Publisher
    .Where(e => !e.IsDeleted)
    .ToListAsync();
Run Code Online (Sandbox Code Playgroud)

也就是说,根据什么ApiUtils.GetMedia(media),您甚至可以对查询执行投影:

var publishers = await _db.Publisher
    .Where(e => !e.IsDeleted)
    .Select(e => new PublisherWithMedia()
    {
        Id = e.Id,
        Name = e.Name,
        Mediae = e.Mediae.Select(m => ApiUtils.GetMedia(m)).ToList()
    };
    .ToListAsync();
Run Code Online (Sandbox Code Playgroud)