前言
IAsyncEnumerable<T>支持返回异步迭代的枚举器,但在.NET 6之前,即使在API中使用了IAsyncEnumerable<T>,它还是使用同步方式输出,首先将结果缓冲到内存中,然后再写入响应中:
[HttpGet]
public IAsyncEnumerable<WeatherForecast> Get()
{
var rng = new Random();
async IAsyncEnumerable<WeatherForecast> streamWeatherForecastsAsync()
{
for (int index = 1; index <= 5; index++)
{
WeatherForecast weatherForecast = new WeatherForecast
{
Date = #34;{DateTime.Now:ss.fff}",
TemperatureC = rng.Next(-20, 55),
Summary = Summaries[rng.Next(Summaries.Length)]
};
await Task.Delay(1000);
yield return weatherForecast;
};
};
return streamWeatherForecastsAsync();
}
现在,在.NET 6中,我们可以使用异步流。
Demo
1.服务端
采用上面相同的代码,可以看到不同的返回效果:
IAsyncEnumerable实例采用异步方式迭代写入响应中,不再阻塞请求调用。
需要注意的是,只有使用System.Text.Json进行序列化时,此功能才会起作用。
2.客户端
同样,我们可以在客户端异步接收响应,示例代码如下:
HttpClient httpClient = new();
var response = await httpClient.GetAsync(
"https://localhost:7211/WeatherForecast",
HttpCompletionOption.ResponseHeadersRead
);
var weatherForecasts = await response.Content
.ReadFromJsonAsync<IAsyncEnumerable<WeatherForecast>>();
await foreach (var weatherForecast in weatherForecasts)
{
Console.WriteLine(#34;[{DateTime.Now:ss.fff}] {weatherForecast.Date}");
}
但是,实际运行效果却是按照同步方式执行的:
这是因为,我们必须使用System.Text.Json提供的专门方法处理异步流反序列化:
var responseStream = await response.Content.ReadAsStreamAsync();
var weatherForecasts = JsonSerializer.DeserializeAsyncEnumerable<WeatherForecast>(responseStream,
new JsonSerializerOptions
{
PropertyNameCaseInsensitive = true,
DefaultBufferSize = 50
});
需要将DefaultBufferSize设置较小的值,保证异步迭代及时返回。
结论
EF Core支持IAsyncEnumerable查询数据:
可以将EF Core从数据库中获取的数据直接使用异步流的方式传输到响应,无需大量内存缓冲数据,提高程序性能。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 xxx@163.com 举报,一经查实,本站将立刻删除。