ASP.NET Core 性能最佳实践(一)

  • A+
所属分类:.NET技术
摘要

这篇文章的主要内容来源于.NET文档,此处翻译前4条内容,其他内容会陆续贴出来

这篇文章的主要内容来源于.NET文档,此处翻译前4条内容,其他内容会陆续贴出来

  1. 尽量使用缓存
  2. 了解”热代码路径”
  3. 避免使用阻塞调用
  4. 返回值使用IEnumerable<T> 或 IAsyncEnumerable<T>

 

尽量使用缓存

详情请查看:ASP.NET Core 中的响应缓存.

 

了解”热代码路径”

”热代码”定义为访问频繁并且耗时较长的代码。”热代码”对性能影响很明细。

 

避免使用阻塞调用

ASP.NET Core 程序应该被设计成同时处理多个请求。异步API使用一个小线程池可以处理上千个并发请求,而不会阻塞。这样请求线程可以去处理其他请求,而不是等待一个长时同步任务完成。

ASP.NET Core 程序的一个常见的性能问题是,阻塞了本该异步执行的调用。很多同步调用会导致线程池饥饿,增大相应时间。

不要像下面这样做:

  • 通过 Task.Wait 或 Task.Result 阻塞异步执行。
  • 在常用代码上使用锁(lock)。ASP.NET Core 程序被设计为并行架构时执行效率最高。
  • 自己定义一个 Task.Run ,然后去await 。由于ASP.NET Core 程序已经是在通用线程池上运行了,调用 Task.Run 是没有必要的,

应该这样做:

  • 异步调用“热代码”。
  • 在访问数据,I/O,长时操作的时候,如果有异步API就尽量使用异步API。不要使用Task.Run将同步API异步执行。
  • 使Controller或Razor Page中的Action异步化。将整个调用栈异步化,以便使用 async/await。

性能分析器 PerfView,可以用来查找频繁加入线程池的线程。

Microsoft-Windows-DotNETRuntime/ThreadPoolWorkerThread/Start 表明一个线程加入了线程池。

 

返回IEnumerable<T> 还是 IAsyncEnumerable<T>

如果Action Result返回IEnumerable<T>,那么序列化器以同步的方式处理集合迭代,这样阻塞调用可能会导致线程池饥饿,为避免同步迭代,可在返回迭代之前调用 ToListAsync()。

从ASP.NET Core 3.0开始,IAsyncEnumerable<T>作为异步枚举,可取代IEnumerable<T>。