有梦想的鱼
写代码我得再认真点,当我最终放下键盘时,我实在不想仍有太多疑惑
1.配置文件的读取

利用Startup类中的configuration读取appsettings.json中的配置节点

{
"Logging": {
  "LogLevel": {
    "Default": "Warning"
  }
},
"option1": "value1_from_json",
"option2": 2,
"subsection": {
  "suboption1": "subvalue1_from_json",
  "Read": [
    "Data Source=.; Database=Customers_New1; User ID=sa; Password=Passw0rd; MultipleActiveResultSets=True",
    "Data Source=ElevenPC; Database=Customers_New2; User ID=sa; Password=Passw0rd; MultipleActiveResultSets=True",
    "Data Source=.; Database=Customers_New3; User ID=sa; Password=Passw0rd; MultipleActiveResultSets=True"
  ]
},
"wizards": [
  {
    "Name": "Gandalf",
    "Age": "1000"
  },
  {
    "Name": "Harry",
    "Age": "17"
  }
],
"AllowedHosts": "*"
}
public IConfiguration Configuration { get; }
public Startup(IConfiguration configuration)
{
    Configuration = configuration;
}
 
//读取Option1节点
this.Configuration["Option1"];
//读取subsection下的suboption1节点
this.Configuration["subsection:suboption1"];
//读取subsection下的Read节点
string[] _SqlConnectionStringRead = this._iConfiguration.GetSection("subsection").GetSection("Read").GetChildren().Select(s => s.Value).ToArray();
2.自带IOC容器
1.基本使用

NuGet安装引用Microsoft.Extensions.DependencyInjection;

实例化容器

注册服务

创建一个System.IServiceProvider提供程序

创建实例

using Microsoft.Extensions.DependencyInjection;
//实例化一个服务容器
IServiceCollection container = new ServiceCollection();
//添加一个瞬时类型到容器
container.AddTransient<Interface,class>();
//创建一个服务provider
IServiceProvider provider = container.BuildServiceProvider();
//从provider中获取接口实例
_Interface  interface = provider.GetService<_Interface>()
2.服务类型

container.AddTransient()最终创建的服务为瞬时生命周期,并且每次创建都是一个全新的实例

_Interface  interface = provider.GetService<_Interface>();
_Interface  interface2 = provider.GetService<_Interface>();
//interface 和 interface2持有的引用不相等,说明不是一个
bool bResult = object.ReferenceEquals(interface, interface2);
//b is false   

container.AddSingleton()最终创建的服务为单例的:全容器都是这一个不变的实例

_Interface  interface = provider.GetService<_Interface>();
_Interface  interface2 = provider.GetService<_Interface>();

//interface 和 interface2持有的引用相等,说明是一个
bool bResult = object.ReferenceEquals(interface, interface2);
//b is True     

container.AddScoped最终创建的服务为请求单例,一个请求代表一个作用域,在同一请求下构建的相同的服务实例只有一个

//创建一个Scope作用域,那么由这个域创建的实例是独立的(相较Scope1)
IServiceScope Scope = provider.CreateScope();
//创建一个Scope1作用域,那么由这个域创建的实例是独立的
IServiceScope Scope1 = provider.CreateScope();

_Interface  interface = Scope.ServiceProvider.GetService<_Interface>();
_Interface  interface2 = Scope1.ServiceProvider.GetService<_Interface>();
//interface 和 interface2持有的引用不相等,因为不是一个作用域
bool bResult = object.ReferenceEquals(interface, interface2);
//b is false    
3.自带IOC容器(IServiceCollection)基础
1.生命周期

瞬时,即时构造,即时销毁

services.AddTransient<ITestServiceA, TestServiceA>();

单例,永远只构造一次

services.AddSingleton<ITestServiceB, TestServiceB>();

作用域单例,一次请求只构造一个

services.AddScoped<ITestServiceC, TestServiceC>(); 
2.实例解析

1.为了方便理解,就不搞一些猫啊狗之类的,当然也比较简陋。我们一共定义了A、B、C、D、E这几个类,代码也比较简单,类与类关联关系如下,

A类构造

B类构造需要A类

C类构造需要B类

D类构造

E类构造需要C类

public TestServiceA()
{
     Console.WriteLine($"{this.GetType().Name}被构造。。。");
}
   
public TestServiceB(ITestServiceA iTestServiceA)
{
     Console.WriteLine($"{this.GetType().Name}被构造。。。");
}
   
public TestServiceC(ITestServiceB iTestServiceB)
{
   Console.WriteLine($"{this.GetType().Name}被构造。。。");
}
   
public TestServiceD()
{
   Console.WriteLine($"{this.GetType().Name}被构造。。。");
}

public TestServiceE(ITestServiceC serviceC)
{
   Console.WriteLine($"{this.GetType().Name}被构造。。。");
}

2.我们首先把他们添加到IOC容器中,然后再使用IOC创建他们的实例

//瞬时
services.AddTransient<ITestServiceA, TestServiceA>();
//单例
services.AddSingleton<ITestServiceB, TestServiceB>();
//作用域单例
services.AddScoped<ITestServiceC, TestServiceC>();
//瞬时
services.AddTransient<ITestServiceD, TestServiceD>();
//瞬时
services.AddTransient<ITestServiceE, TestServiceE>();

第一次请求解析

1.开始构造A类,因为是瞬时生命周期那么第一次请求就会被构造,然后销毁

2.在创建B类时需要A类,那么会首先构造一个A类(因为A是瞬时,上一次已经被销毁,所以是一个全新的A),然后构造B类为单例

3.开始创建C,在构造时需要B类,因为B类全局单例已经存在,那就会直接构造C作用域单例

4.直接创建瞬时生命周期D

5.开始创建瞬时生命周期E类,构造时需要C类,因为C类为作用域单例,那么就会直接构造E类为瞬时

第一次请求结论

1.因为瞬时所以输出A被构造

2.输出A被构造B被构造,此时B全局单例

3.C被创建时需要B,但是B是全局单例,所以不会被构造,所以只输出C被构造

4.输出D被构造

5.C为作用域单例,B为全局单例,所以只输出E被构造

第二次请求解析

1.开始构造A类,因为A瞬时生命周期那么第二次请求就会重新被构造,然后销毁

2.在构造B类时因为在第一次请求时已经构造为单例,所以不再被构造,直接使用

3.开始重新构造C,因为C在第二次请求为新的作用域,在构造时需要B类,因为B类全局单例,那直接构造C为当前新的作用域单例

4.直接构造瞬时生命周期D类

5.开始构造E类,构造式需要C类,因为C类为作用域单例,那么就会直接构造E类为瞬时

第二次请求结论

1.输出A被构造

2.输出C被构造

3.输出D被构造

4.输出E被构造

3.手动创建实例

​ 有人会说,如果想通过构造函数全部自动注入,我们可以使用服务Procider自己在使用时创建对应自己需要的服务实例

1.引入命名空间using Microsoft.Extensions.DependencyInjection;

2.首先注入 IServiceProvider serviceProvider,利用serviceProcider.GetService<ITestServiceA>();生成需要的实例

4.AutoFac使用
1.Autofac容器替换自带容器

1.下载Autofac.Extensions.DependencyInjectionAutofac

2.在Program网站启动时替换默认IOC工厂实例为AutofacUseServiceProviderFactory(new AutofacServiceProviderFactory())

3.在Startup类中添加方法

//返回值为IServiceProvider
public IServiceProvider ConfigureContainer(ContainerBuilder services)
{
services.Configure<CookiePolicyOptions>(options =>
{
   options.CheckConsentNeeded = context => true;
   options.MinimumSameSitePolicy = SameSiteMode.None;
});
services.AddSession();
services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_2);
//services.RegisterType<TestServiceE>().As<ITestServiceE>().SingleInstance();
services.RegisterModule<CustomMoudle>();

}
2.批量注册模块
public class CustomAutofacModule : Module
{
    //重新load 添加注册服务
    //可以实现实例控制
    //接口约束
   protected override void Load(ContainerBuilder containerBuilder)
   {
       containerBuilder.Register(c => new CustomAutofacAop());
      
      //设置为属性注入
       containerBuilder.RegisterType<TestServiceA>().As<ITestServiceA>().SingleInstance().PropertiesAutowired();
       
       containerBuilder.RegisterType<TestServiceC>().As<ITestServiceC>();
       containerBuilder.RegisterType<TestServiceB>().As<ITestServiceB>();
       containerBuilder.RegisterType<TestServiceD>().As<ITestServiceD>();

       containerBuilder.RegisterType<A>().As<IA>().EnableInterfaceInterceptors();
   }
}
3.利用AutoFac实现AOP

1.NuGet安装引用Autofac.Extras.DynamicProxy

2.引用using Autofac.Extras.DynamicProxy;

3.创建类库继承自IInterceptor实现接口

4.自定义的Autofac的AOP扩展

 //注册,模块
public class CustomAutofacModule : Module
{
     //重新load 添加注册服务
     //可以实现实例控制
     //接口约束
    protected override void Load(ContainerBuilder containerBuilder)
    {
        //注册AOP
        containerBuilder.Register(c => new CustomAutofacAop());
        //开启aop扩展
        containerBuilder.RegisterType<A>().As<IA>().EnableInterfaceInterceptors();
    }
}

public class CustomAutofacAop : IInterceptor
{
   public void Intercept(IInvocation invocation)
   {
       Console.WriteLine($"invocation.Methond={invocation.Method}");
       Console.WriteLine($"invocation.Arguments={string.Join(",", invocation.Arguments)}");

       invocation.Proceed(); //继续执行

       Console.WriteLine($"方法{invocation.Method}执行完成了");
   }
}

posted on 2022-01-20 13:39  有梦想的鱼i  阅读(1689)  评论(0编辑  收藏  举报