草庐IT

ASP.NET Core依赖注入系统学习教程:针对服务注册的验证

坚持 2023-03-28 原文

1.避免Scoped模式注册的服务变成Singleton模式

当提供一个生命周期模式为Singleton的服务实例时,如果发现该服务中还依赖生命周期模式为Scoped的服务实例(Scoped服务实例将被一个Singleton服务实例所引用),那么这个被依赖的Scoped服务实例最终会成为一个Singleton模式的服务实例。这是因为提供Singleton服务的容器是根容器,Scoped服务间接的被根容器所创建提供了,如果Scoped服务由子容器进行提供,那么Singleton和Scoped这两种生命周期模式才会产生差别。

在ASP.NET Core应用中,将某个服务注册的生命周期设置为Scoped,其意图是希望Scoped服务实例的创建和释放是作用于某个HTTP请求范围内的。如果不注意,将Scoped服务实例引用到了Singleton服务实例中,对于这种情况Scoped和Singleton的服务实例没有区别的。这样的Scoped服务实例直到应用关闭才会被释放,这无疑违背我们使用Scoped模式的初衷。这种“混淆”如果没有察觉到,可能会在实际的应用中造成难以估量的后果,例如在Singleton服务中引用的Scoped服务是一个数据库连接对象,这会导致数据库长时间连接没有及时释放,从而导致程序出现异常。

为了避免Scoped模式注册的服务“隐式”的变成Singleton模式的服务带来的风险,.NET Core为我们提供了一种验证方式来规避这样的行为,这个方式就是将ServiceProviderOptions配置对象的ValidateScopes属性设置为True。

当开启了这个验证后,依赖注入框架则会对注册Scoped模式的服务进行检查,确保不会出现如下情况:

  • 有根容器去提供Scoped的服务实例;
  • Singleton服务中存在对Scoped服务的依赖;

一旦开启针对Scoped模式服务的注册验证,如果存在以上的两种情况,那么程序启动时会抛出异常。

ValidateScopes的值在开发环境下默认值是为True,为了确保在生产环境或其他环境始终开启验证,我们可以在Program类的CreateHostBuilder方法中配置ServiceProviderOptions对象。

该验证方式被官方称作为“服务范围”的验证,并建议应用程序开启此验证,以确保我们注册Scoped模式的服务仅作用于某个服务范围,而不会“悄悄地”演变成作用于整个应用程序范围的Singleton模式。


2.验证服务注册是否能够提供相应的实例

依赖注入框架中进行服务注册的信息一般都存放于ServiceDescriptor的对象中,而容器对象就是根据ServiceDescriptor对象中的注册信息进行服务实例的提供。ServiceProviderOptions配置类型除了用于针对“服务范围”的验证ValidateScopes属性之外,还有一个ValidateOnBuild属性。如果将该属性设置为True,这就意味着容器对象在构建时,会对每个ServiceDescriptor对象中的注册信息实施有效性验证,如果服务注册信息不能提供出对应的实例则会抛出异常。

使用ValidateOnBuild属性进行验证的目的是因为,往往有些服务能够正常注册但不代表,容器能够根据注册信息成功的提供实例。下面我将通过一个代码示例来印证这一情况,并演示使用ValidateOnBuild属性进行验证的方式。

 1 using Microsoft.Extensions.DependencyInjection;
 2 using System;
 3 using System.Diagnostics;
 4 
 5 namespace ConsoleApp1
 6 {
 7 
 8     public interface IFooBar { }
 9     public class FooBar : IFooBar
10     {
11         private FooBar() { }
12     }
13 
14     internal class Program
15     {
16         static void Main(string[] args)
17         {
18             Console.WriteLine("ValidateOnBuild的值为True:");
19             BuildServiceProvider(true);
20             Console.WriteLine();
21 
22             Console.WriteLine("ValidateOnBuild的值为False:");
23             BuildServiceProvider(false);
24 
25         } // END Main()
26 
27         static void BuildServiceProvider(bool validateOnBuild)
28         {
29             try
30             {
31                 var options = new ServiceProviderOptions { ValidateOnBuild = validateOnBuild };
32              var provider=   new ServiceCollection()
33                     .AddSingleton<IFooBar, FooBar>()
34                     .BuildServiceProvider(options);
35                 Console.WriteLine($"程序运行正常;");
36             }
37             catch (Exception e)
38             {
39                 Console.WriteLine($"程序出现异常;异常信息:{e.Message};");
40             }
41 
42         } // END BuildServiceProvider()
43 
44 
45     }
46 }

在上面的代码示例中,服务注册时指定的实现类型为FooBar,而该类型中唯一的构造函数是一个私有的。我们都知道创建对象的实例,则必须要调用对象类型的构造函数,而FooBar的构造函数是私有的,无法对外界所调用,则也意味着无法创建相应的实例。

运行上面的代码示例后我们会发现,对于ValidateOnBuild属性值设置为False的情况,程序可以正常的执行服务注册的方法,这种现象隐瞒了注册时的错误(类型中存在私有构造函数),这种做法显然会对后续使用到FooBar对象的程序功能带来影响。而对于ValidateOnBuild属性值设置为True的情况,程序则直接抛出了异常并且给出了详细的错误信息,对于这种做法我们可以更好的规避服务注册时所产生的风险。

 

本示例代码是在“服务定位器模式”下配置的ValidateOnBuild属性,这是为了更好的演示。当然,我们在实际的开发中通常是依赖注入的形式,那么相应的方式是通过在CreateHostBuilder方法中ServiceProviderOptions对象进行配置,如下图:

 


3.总结

本文介绍的关于“针对服务注册的验证”的主题,实际上就是介绍ServiceProviderOptions类型中ValidateScopesValidateOnBuild属性的使用场景和方式。VailedateScopes属性主要确保我们注册的Scoped服务不会在某些情况下变成Singleton服务,ValidateOnBuild属性主要用于验证服务注册信息是否能成功的提供出对应的服务示例。

有关ASP.NET Core依赖注入系统学习教程:针对服务注册的验证的更多相关文章

  1. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  2. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

    我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

  3. ruby-on-rails - 启动 Rails 服务器时 ImageMagick 的警告 - 2

    最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru

  4. ruby-on-rails - 在 ruby​​ .gemspec 文件中,如何指定依赖项的多个版本? - 2

    我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这

  5. ruby-on-rails - s3_direct_upload 在生产服务器中不工作 - 2

    在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo

  6. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

    我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

  7. ruby - 用 Ruby 编写一个简单的网络服务器 - 2

    我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b

  8. ruby-on-rails - 在 Rails 中调试生产服务器 - 2

    您如何在Rails中的实时服务器上进行有效调试,无论是在测试版/生产服务器上?我试过直接在服务器上修改文件,然后重启应用,但是修改好像没有生效,或者需要很长时间(缓存?)我也试过在本地做“脚本/服务器生产”,但是那很慢另一种选择是编码和部署,但效率很低。有人对他们如何有效地做到这一点有任何见解吗? 最佳答案 我会回答你的问题,即使我不同意这种热修补服务器代码的方式:)首先,你真的确定你已经重启了服务器吗?您可以通过跟踪日志文件来检查它。您更改的代码显示的View可能会被缓存。缓存页面位于tmp/cache文件夹下。您可以尝试手动删除

  9. 电脑0x0000001A蓝屏错误怎么U盘重装系统教学 - 2

      电脑0x0000001A蓝屏错误怎么U盘重装系统教学分享。有用户电脑开机之后遇到了系统蓝屏的情况。系统蓝屏问题很多时候都是系统bug,只有通过重装系统来进行解决。那么蓝屏问题如何通过U盘重装新系统来解决呢?来看看以下的详细操作方法教学吧。  准备工作:  1、U盘一个(尽量使用8G以上的U盘)。  2、一台正常联网可使用的电脑。  3、ghost或ISO系统镜像文件(Win10系统下载_Win10专业版_windows10正式版下载-系统之家)。  4、在本页面下载U盘启动盘制作工具:系统之家U盘启动工具。  U盘启动盘制作步骤:  注意:制作期间,U盘会被格式化,因此U盘中的重要文件请注

  10. 【鸿蒙应用开发系列】- 获取系统设备信息以及版本API兼容调用方式 - 2

    在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList​()Obt

随机推荐