为什么要避免同步读取
ASP.NET Core 中的所有 I/O 操作都是异步的。服务器实现了 Stream
接口,该接口同时具备同步和异步的方法。
在进行 I/O 操作时,应优先使用异步方法,以避免阻塞线程池的线程。
如果阻塞了线程池线程,可能会导致服务器无法处理更多请求,造成急剧性性能下降。
尤其是当客户端上传速度缓慢时 ,同步读取将阻塞线程直到整个请求体被全部读取完成。
如何避免同步读取
错误的做法
以下代码示例使用了同步方法 ReadToEnd
,导致线程被阻塞:
csharp
public class BadStreamReaderController : Controller
{
[HttpGet("/contoso")]
public ActionResult<ContosoData> Get()
{
var json = new StreamReader(Request.Body).ReadToEnd();
return JsonSerializer.Deserialize<ContosoData>(json);
}
}
在这段代码中,Get
方法将整个 HTTP 请求体同步读入内存。如果客户端上传速度缓慢,应用程序将阻塞在这个读取操作上,导致效率下降。
正确的做法
使用异步方法 ReadToEndAsync
,可以避免阻塞线程:
csharp
public class GoodStreamReaderController : Controller
{
[HttpGet("/contoso")]
public async Task<ActionResult<ContosoData>> Get()
{
var json = await new StreamReader(Request.Body).ReadToEndAsync();
return JsonSerializer.Deserialize<ContosoData>(json);
}
}
这段代码使用异步读取方法,在读取过程中不会阻塞线程,可以提升性能和响应速度。
读取表单数据时的注意事项
错误的做法
使用 HttpContext.Request.Form
,会在内部执行同步读取,导致线程被阻塞:
csharp
public class BadReadController : Controller
{
[HttpPost("/form-body")]
public IActionResult Post()
{
var form = HttpContext.Request.Form;
Process(form["id"], form["name"]);
return Accepted();
}
}
正确的做法
使用 ReadFormAsync
,进行异步读取:
csharp
public class GoodReadController : Controller
{
[HttpPost("/form-body")]
public async Task<IActionResult> Post()
{
var form = await HttpContext.Request.ReadFormAsync();
Process(form["id"], form["name"]);
return Accepted();
}
}
这种做法使用异步方式读取表单数据,能有效避免阻塞线程池资源。
结论
在 ASP.NET Core 开发中,应符合框架的异步操作模式,避免使用同步方法读取 HTTP 请求文本。
这样可以有效地提升应用程序的性能和响应速度,避免因阻塞导致的急剧性性能下降。