ASP.NET Core依赖注入最佳实践,提示技巧

分享翻译一篇Abp框架作者(Halil İbrahim Kalkan)关于ASP.NET Core依赖注入的博文.

640?wx_fmt=png

在本文中,我将分享我在ASP.NET Core应用程序中使用依赖注入的经验和建议.

这些原则背后的目的是:

  1. 有效地设计服务及其依赖关系

  2. 防止多线程问题

  3. 防止内存泄漏

  4. 防止潜在的错误

本文假设你已经熟悉基本的ASP.NET Core以及依赖注入. 如果没有的话,请首先阅读ASP.NET核心依赖注入文档.
ASP.NET Core 依赖注入文档

构造函数注入

构造函数注入用在服务的构造函数上声明和获取依赖服务.
例如:

640?wx_fmt=png

ProductService在构造函数中将IProductRepository注入为依赖项,然后在Delete方法中使用它.

属性注入

ASP.NET Core的标准依赖注入容器不支持属性注入,但是你可以使用其它支持属性注入的IOC容器.
例如:

640?wx_fmt=png

ProductService具有公开的Logger属性. 依赖注入容器可以自动设置Logger(前提是ILogger之前注册到DI容器中).

建议做法

  1. 仅对可选依赖项使用属性注入。这意味着你的服务可以脱离这些依赖能正常工作.

  2. 尽可能得使用Null对象模式(如本例所示Logger = NullLogger<ProductService>.Instance;), 不然就需要在使用依赖项时始终做空引用的检查.

服务定位器

服务定位器模式是获取依赖服务的另一种方式.
例如:

640?wx_fmt=png

ProductService服务注入IServiceProvider并使用它来解析其依赖,如果欲解析的依赖未注册GetRequiredService会抛出异常,GetService只返回NULL.

在构造函数中解析的依赖,它们将会在服务被释放的时候释放,因此你不需要关心在构造函数中解析的服务释放/处置(release/dispose),这点同样适用于构造函数注入和属性注入.

建议做法

  1. 如果在开发过程中已知依赖的服务尽可能不使用服务定位器模式, 因为它使依赖关系含糊不清,这意味着在创建服务实例时无法获得依赖关系,特别是在单元测试中需要模拟服务的依赖性尤为重要.

  2. 尽可能在构造函数中解析所有的依赖服务,在服务的方法中解析服务会使你的应用程序更加的复杂且容易出错.我将在下一节中介绍在服务方法中解析依赖服务

服务生命周期

ASP.NET Core下依赖注入中有三种服务生命周期:

  1. Transient,每次注入或请求时都会创建转瞬即逝的服务.

  2. Scoped,是按范围创建的,在Web应用程序中,每个Web请求都会创建一个新的独立服务范围.这意味着服务根据每个Web请求创建.

  3. Singleton,每个DI容器创建一个单例服务,这通常意味着它们在每个应用程序只创建一次,然后用于整个应用程序生命周期.

DI容器自动跟踪所有已解析的服务,服务在其生命周期结束时被释放/处置(release/dispose)

  1. 如果服务具有依赖关系,则它们的依赖的服务也会自动释放/处置(release/dispose)

  2. 如果服务实现IDisposable接口,则在服务被释放时自动调用Dispose方法.

建议做法

  1. 尽可能将你的服务生命周期注册为Transient,因为设计Transient服务很简单,你通常不关心多线程和内存泄漏,该服务的寿命很短.

  2. 请谨慎使用Scoped生命周期的服务,因为如果你创建子服务作用域或从非Web应用程序使用这些服务,则可能会非常棘手.

  3. 小心使用Singleton生命周期的服务,这种情况你需要处理多线程和潜在的内存泄漏问题.

  4. 不要在Singleton生命周期的服务中依赖TransientScoped生命周期的服务.因为Transient生命周期的服务注入到Singleton生命周期的服务时变为单例实例,如果Transient生命周期的服务没有对此种情况特意设计过,则可能导致问题. ASP.NET Core默认DI容器会对这种情况抛出异常.

在服务方法中解析依赖服务

在某些情况下你可能需要在服务方法中解析其他服务.在这种情况下,请确保在使用后及时释放解析得服务,确保这一点的最佳方法是创建Scoped服务.
例如:

640?wx_fmt=png

PriceCalculator在构造函数中注入IServiceProvider服务,并赋值给_serviceProvider属性. 然后在PriceCalculator的Calculate方法中使用它来创建子服务范围。 它使用scope.ServiceProvider来解析服务,而不是注入的_serviceProvider实例。 因此从范围中解析的所有服务都将在using语句的末尾自动释放/处置(release/dispose)

建议做法

  1. 如果要在方法体中解析服务,请始终创建子服务范围以确保正确的释放已解析的服务.

  2. 如果将IServiceProvider作为方法的参数,那么你可以直接从中解析服务而无需关心释放/处置(release/dispose). 创建/管理服务范围是调用方法的代码的责任. 遵循这一原则使你的代码更清晰.

  3. 不要引用解析到的服务,不然它可能会导致内存泄漏或者在你以后使用对象引用时可能访问已处置的(dispose)服务(除非服务是单例)

单例服务(Singleton Services)

单例服务通常用于保持应用程序状态. 缓存服务是应用程序状态的一个很好的例子.
例如:

640?wx_fmt=png

FileService缓存文件内容以减少磁盘读取. 此服务应注册为Singleton,否则缓存将无法按预期工作.

建议做法

  1. 如果服务需要保持状态,则应以线程安全的方式访问该状态.因为所有请求同时使用相同的服务实例.我使用ConcurrentDictionary而不是Dictionary来确保线程安全.

  2. 不要在单例服务中使用Scoped生命周期或Transient生命周期的服务.因为临时服务可能不是设计为线程安全.如果必须使用它们那么在使用这些服务时请注意多线程问题(例如使用锁).

  3. 内存泄漏通常由单例服务引起.它们在应用程序结束前不会被释放/处置(release/dispose). 因此如果他们实例化的类(或注入)但不释放/处置(release/dispose).它们,它们也将留在内存中直到应用程序结束. 确保在正确的时间释放/处置(released/disposed)它们。 请参阅上面的在方法中的解析服务内容.

  4. 如果缓存数据(本示例中的文件内容),则应创建一种机制,以便在原始数据源更改时更新/使缓存的数据无效(当上面示例中磁盘上的缓存文件发生更改时).

范围服务(Scoped Services)

Scoped生命周期的服务乍一看似乎是存储每个Web请求数据的良好候选者.因为ASP.NET Core会为每个Web请求创建一个服务范围. 因此,如果你将服务注册为作用域则可以在Web请求期间共享该服务.
例如:

640?wx_fmt=png

如果将RequestItemsService注册为Scoped并将其注入两个不同的服务,则可以获取从另一个服务添加的项,因为它们将共享相同的RequestItemsService实例.这就是我们对Scoped生命周期服务的期望.

但是...事实可能并不总是那样. 如果你创建子服务范围并从子范围解析RequestItemsService,那么你将获得RequestItemsService的新实例,它将无法按预期工作.因此,作用域服务并不总是表示每个Web请求的实例。

你可能认为你没有犯这样一个明显的错误(在子范围内解析服务). 情况可能不那么简单. 如果你的服务之间存在大的依赖关系,则无法知道是否有人创建了子范围并解析了注入另一个服务的服务.最终注入了作用域服务.

建议做法

  1. Scoped生命周期的服务可以被认为是在Web请求中由太多服务注入的优化.因此,所有这些服务将在同一Web请求期间使用该服务的单个实例.

  2. Scoped生命周期的服务不需要设计为线程安全的. 因为它们通常应由单个Web请求/线程使用.但是...在这种情况下,你不应该在不同的线程之间共享Scoped生命周期服务!

  3. 如果你设计Scoped生命周期服务以在Web请求中的其他服务之间共享数据,请务必小心(如上所述). 你可以将每个Web请求数据存储在HttpContext中(注入IHttpContextAccessor以访问它),这是更安全的方式. HttpContext的生命周期不是作用域. 实际上它根本没有注册到DI(这就是为什么你不注入它,而是注入IHttpContextAccessor). HttpContextAccessor使用AsyncLocal实现在Web请求期间共享相同的HttpContext.

结论

依赖注入起初看起来很简单,但是如果你不遵循一些严格的原则,就会存在潜在的多线程和内存泄漏问题. 我根据自己在ASP.NET Boilerplate框架开发过程中的经验分享了一些很好的原则.

相关文章:

  • C#中字段、属性、只读、构造函数赋值、反射赋值的相关

原文地址:https://www.cnblogs.com/realmaliming/p/9467601.html


.NET社区新闻,深度好文,欢迎访问公众号文章汇总 http://www.csharpkit.com 

640?wx_fmt=jpeg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/320413.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【匈牙利算法】指引(jzoj 2319)

指引 jzoj 2319 题目大意&#xff1a; 在平面上有n个人和出口&#xff0c;一个出口只能让一个人进&#xff0c;每个人只能向右向上走&#xff0c;问最多让多少个人到出口 输入样例&#xff1a; 6 3 2 0 3 1 1 3 4 2 0 4 5 5输出样例&#xff1a; 2解题思路&#xff1a; …

Codeforces1142D

Codeforces1142D 做法&#xff1a;构建一个可以识别出合法串的自动机&#xff0c;然后就可以想办法在上边 dp 出答案。 首先&#xff0c;按照最直观的思路画一画这个自动机&#xff0c;找到每一个状态s如何推出它的后继t&#xff0c;然后通过状态的转移方式&#xff0c;找到等价…

P3620-[APIO/CTSC2007]数据备份【贪心,堆,链表】

正题 题目链接:https://www.luogu.com.cn/problem/P3620 题目大意 一条线上有nnn个位置&#xff0c;选出kkk对使得它们的距离差之和最小。 解题思路 因为一定是连接相邻的最优&#xff0c;那么可以在差分数组上做&#xff0c;相当于我们在一个差分数组上选择一些不相邻的数使…

Maximize The Beautiful Value

传送 时间限制&#xff1a;C/C 2秒&#xff0c;其他语言4秒 空间限制&#xff1a;C/C 131072K&#xff0c;其他语言262144K 64bit IO Format:%lld 题目描述 Today HH finds a non-decreasing sequence(a1,a2…an,ai≤ai1), he thinks it’s not beautiful so he wants to make …

纪中C组模拟赛总结(2019.9.7)

成绩&#xff1a; 注&#xff1a;rankrankrank是有算其他dalaodalaodalao的 hky,wjj,lthhky,wjj,lthhky,wjj,lth三位dalaodalaodalao竟不屑于交题 rankrankranknamenamenamescorescorescoreT1T1T1T2T2T2T3T3T3T4T4T4888lyflyflyf110110110101010000100100100000242424fyfyfy45…

MEF 插件式开发 - DotNetCore 初体验

背景叙述在传统的基于 .Net Framework 框架下进行的 MEF 开发&#xff0c;大多是使用 MEF 1&#xff0c;对应的命名空间是 System.ComponentModel.Composition。在 DotNet Core 中&#xff0c;微软为了伟大的跨平台策略&#xff0c;引入了 MEF 2&#xff0c;其对应的命名空间是…

反向传播算法学习笔记

反向传播算法(Back propagation) 目的及思想 我们现在有一堆输入&#xff0c;我们希望能有一个网络&#xff0c;使得通过这个网络的构成的映射关系满足我们的期待。也就是说&#xff0c;我们在解决这个问题之前先假设&#xff0c;这种映射可以用网络的模型来比较好的描述。为什…

AT1219-歴史の研究(历史研究)【回滚莫队】

正题 题目链接:https://www.luogu.com.cn/problem/AT1219 题目大意 nnn个数字&#xff0c;mmm次询问一个区间内ti∗it_i*iti​∗i的最大值&#xff0c;tit_iti​即区间内iii的出现次数。 解题思路 用回滚莫队的思想&#xff0c;对于在不同块中的询问&#xff0c;我们把左端点…

求树的直径

欢迎来踩本人博客 树的直径&#xff1a; 就是树上最长路 方法 &#xff1a; 求两边DFS即可 步骤&#xff1a; 1.从任意一点进行dfs&#xff0c;然后找到一个最长路径&#xff0c;记录最远点u 2.然后从u再进行dfs&#xff0c;找最长路径&#xff0c;记录一点v。 &#xff08;u&…

【暴力】MSWORLD

MSWORLD 题目大意&#xff1a; 在一个图上有n个点&#xff0c;现在问你最远的两个点的直线距离的平方是多少 输入样例 4 0 0 0 1 1 1 1 0输出样例 2样例解释&#xff1a; 农场1&#xff08;0,0&#xff09;和农场3&#xff08;1,1&#xff09;的距离为 2的开方。 数据范…

微软技术直通车(第三期) 之 人工智能

编者&#xff1a;有幸本周在北京&#xff0c;大家有空来现场面基。微软技术直通车本系列活动密切关注微软及周边相关技术。以微软云计算和相关产品为依托&#xff0c;涉及云计算、数据处理、开发工具、商用软件、物联网、人工智能等前沿科技。系列活动邀请微软技术专家、一线开…

SDOI2018 物理实验

SDOI2018 物理实验 题意&#xff1a;二维平面上有一条直线&#xff0c;直线上放置了一个激光发射器&#xff0c;会向导轨两侧沿导轨垂直方向发射宽度为 L 的激光束。平面上还有 n 条线段&#xff0c;并且线段和线段、线段和直线之间都没有公共点&#xff0c;线段和直线的夹角不…

P2485-[SDOI2011]计算器【BSGS,exgcd,快速幂】

正题 题目链接:https://www.luogu.com.cn/problem/P2485 题目大意 给出a,b,pa,b,pa,b,p要求一下一种 ab%pa^b\% pab%p的值ax≡b(modp)ax\equiv b(\mod p)ax≡b(modp)的最小非负整数解ax≡b(modp)a^x\equiv b(\mod p)ax≡b(modp)的最小非负整数解 解题思路 一道缝合题 第一个…

幸运数字Ⅱ

牛客网 时间限制&#xff1a;C/C 1秒&#xff0c;其他语言2秒 空间限制&#xff1a;C/C 262144K&#xff0c;其他语言524288K 64bit IO Format: %lld 题目描述 定义一个数字为幸运数字当且仅当它的所有数位都是4或者7。 比如说&#xff0c;47、744、4都是幸运数字而5、17、467都…

【背包】SMRTFUN

SMRTFUN 题目大意&#xff1a; 有n件物品&#xff0c;每件物品有各自的a值和b值&#xff0c;现在让你选一些物品&#xff0c;在a、b都不是负数的情况下&#xff0c;使a、b值之和最大 输入样例 5 -5 7 8 -6 6 -3 2 1 -8 -5输出样例 8样例说明 选择第1&#xff0c;3和4号牛…

Visual Studio 2017 15.8 正式发布,测试速度提高 82%

Visual Studio 2017 15.8 版本已正式发布&#xff1a;发行说明&#xff1a;https://docs.microsoft.com/zh-cn/visualstudio/releasenotes/vs2017-relnotes#15.8下载地址&#xff1a;https://visualstudio.microsoft.com/downloads/安装现可选择在开始安装之前下载所有安装文件…

Codeforces 1176F

Codeforces 1176F 题目 题意&#xff1a;T组物品&#xff0c;按顺序选一个一个选&#xff0c;物品首先要满足组间的相对顺序&#xff0c;每个物品有价值和体积&#xff0c;每组选择的体积不能超过3&#xff0c;组内的选择物品的顺序可以调整&#xff0c;在总的物品的顺序中&…

jzoj5702-[gdoi2018day2]滑稽子图【树形dp,二项式定理】

正题 题目大意 nnn个点的一棵树&#xff0c;定义f(S)f(S)f(S)表示点集SSS的生成子图中的边数量。 求∑S∈Vf(S)k\sum_{S\in V}f(S)^kS∈V∑​f(S)k 解题思路 因为kkk很小&#xff0c;所以可以考虑一下二项式拆解&#xff0c;我们需要快速的计算出(ab)k(ab)^k(ab)k&#xff0c…

【每日一题】4月9日题目精讲 Running Median

文章目录题目&#xff1a;题意&#xff1a;题解一&#xff1a;题解二&#xff1a;题目&#xff1a; –>链接<— 时间限制&#xff1a;C/C 5秒&#xff0c;其他语言10秒 空间限制&#xff1a;C/C 65536K&#xff0c;其他语言131072K 64bit IO Format:%lld 题目描述 For t…

【结论】区间和的和

区间和的和 题目大意&#xff1a; 给出一个数组&#xff0c;求出所有区间和的总和 输入样例&#xff1a; 3 1 2 3输出样例&#xff1a; 20数据范围&#xff1a; 对于30%的数据&#xff1a;1⩽n⩽1001\leqslant n\leqslant 1001⩽n⩽100 对于50%的数据&#xff1a;1⩽n⩽1…