重新整理 .net core 实践篇————依赖注入应用[二]
前言
这里介绍一下.net core的依赖注入框架,其中其代码原理在我的另一个整理<<重新整理 1400篇>>中已经写了,故而专门整理应用这一块。
以下只是个人整理,如有问题,望请指点。
正文
为什么有这个依赖注入呢?
假设人们面临这样一个问题。
比如说一个人做飞机去北京。那么人和飞机有什么关系呢?人和北京有什么关系呢?
假设有3个主体类,一个是人一个是飞机一个是北京。他们之间到底应该怎么组织呢?
先说人和北京的关系,人要去北京,那么飞机就是一个交通工具。是的,只是一个交通工具,那么人和飞机就没有直接的关系,而是人通过交通工具去了北京。
同样北京也可以抽象处理,北京是一个地点。那么这个语句就变成了"人通过交通工具去了某个地方",这个交通工具可以是飞机可以是高铁,未来可以是火箭。地点同样如此。
如果抓住这个词,“未来”,那么其扩展性就很强了,不仅现在拥有的还代表以后。就像我们的usb接口一样,未来可能有新的设备只要适配了usb接口,那么我主机里面的东西将不用动任何东西就可以运行新的设备。
因为我的主机没有动任何东西,那么是不是代表我的稳定性?所以依赖注入不仅可以确保我们代码的可维护性和可扩展性,最重要的是稳定性(当然了稳定性是可维护性的一种),而代码最重要的就是稳定性。
在asp.net core 的整个架构中,依赖注入框架提供了对象创建和生命周期管理的核心能力,各个组件相互协作,可以说是整个框架的核心了。
那么这里有一个关键词生命周期,将会伴随着整篇文章的核心。
.net core 的依赖实现主要是通过 Microsoft.Extensions.DependencyInjection;
其这个包,是实现了Microsoft.Extensions.DependencyInjection.Abstrations;
Microsoft.Extensions.DependencyInjection.Abstrations是接口的定义,Microsoft.Extensions.DependencyInjection 是接口的实现。
那么这里就可以想象到既然提供了接口,那么一般来说就有第三方实现,后面会介绍第三方实现。
依赖注入的核心由下面几个部门组成。
IServiceCollection 负责服务的注册
ServiceDescriptor 每一个服务注册时候的信息
IServiceProvider 具体的容器
IServiceScope 容器的子容器的生命周期
那么什么是生命周期呢?
单例模式
作用域模式
瞬时模式
那么这个怎么理解呢?首先建立起这样一个模型。
这个模型是什么意思呢? 首先我们创建单例模式产生的对象会存在root中,如果作用域模式那么创建的对象将会在某个作用域中,这个作用域是自己指定的。
而瞬时模式呢,就是不存储在任何一个作用域中。
当我们通过这个依赖注入服务创建对象的时候,如果是单例模式,那么会这么干,查找root容器中有没有该注册信息的实例,如果有就直接返回,如果没有那么创建存储在root容器中。
如果是作用域模式,那么会去找当前作用域有没有该注册信息的实例,如果有那么直接返回,如果没有那么存储在当前作用域的容器中。
同样,如果作用域模式,那么该作用域的容器消失了,里面存储的东西就没了,那么自然通过该作用域创建的对象就消失了。
那么就用代码来实验一下。
有三个类:
里面的内容就是一个接口,然后一个具体的类,这里演示其中一个。
public interface IMySingletonService
{
}
public class MySingletonService: IMySingletonService
{
}
然后再configureServices 中进行注册:
services.AddSingleton<IMySingletonService, MySingletonService>();
services.AddScoped<IMyScopedService, MyScopedService>();
services.AddTransient<IMyTransientService, MyTransientService>();
然后写一个http的get方法。
[HttpGet]
public int GetService([FromServices] IMySingletonService mySingleton1,
[FromServices] IMySingletonService mySingleton2,
[FromServices]IMyScopedService myScoped1,
[FromServices] IMyScopedService myScoped2,
[FromServices] IMyTransientService myTransient1,
[FromServices] IMyTransientService myTransient2)
{
Console.WriteLine($"singleton1:{mySingleton1.GetHashCode()}");
Console.WriteLine($"singleton1:{mySingleton2.GetHashCode()}");
Console.WriteLine($"myScoped1:{myScoped1.GetHashCode()}");
Console.WriteLine($"myScoped2:{myScoped2.GetHashCode()}");
Console.WriteLine($"myTransient1:{myTransient1.GetHashCode()}");
Console.WriteLine($"myTransient2:{myTransient2.GetHashCode()}");
return 1;
}
这里通过hashcode 来验证是否两个对象是否相等。
截图如下:
然后我们再访问一次这个接口。
上述可以得出,每一个http请求将会在同一个子容器中,且同一子容器获取的注册对象相同。而单例每次都相同。
transient 则是每次获取到不同的对象。
好了,现在生命周期明了了,下面看一下注册的方式。
我们通过这种方式进行注册:
services.AddSingleton<IMySingletonService, MySingletonService>();
那么是否还有其他方式?
还可以这样:
services.AddSingleton<IMySingletonService>(new MySingletonService());
services.AddSingleton<IMySingletonService>(ServiceProvider =>
{
return new MySingletonService();
});
为什么需要这样呢?道理也是很简单的,因为我们可能需要一些自己定制的动态参数啊。
比如说:
services.AddSingleton<IMySingletonService>(ServiceProvider =>
{
return new MySingletonService();
});
这种工厂模式,完全可以通过其他的注册服务(ServiceProvider)的参数来实例化我们的MySingletonService。
还有值得注意的是,因为我们可以多次注入。那么我们按道理可以获取到注册的全部。可以通过这种方式。
比如说:
services.AddSingleton<IMySingletonService, MySingletonService>();
services.AddSingleton<IMySingletonService>(new MySingletonService());
services.AddSingleton<IMySingletonService>(ServiceProvider =>
{
return new MySingletonService();
});
通过AddSingleton调用了3次。
[HttpGet]
public int GetService([FromServices] IEnumerable<IMySingletonService> mySingleton1)
{
foreach (var item in mySingleton1)
{
Console.WriteLine(item.GetHashCode());
}
return 1;
}
然后获取一下。
发现可以一对多了,分别获取到3个不同注册的。
那么如果我们的逻辑比较复杂,可能会多次调用到怎么破?
services.AddSingleton<IMySingletonService, MySingletonService>();
services.TryAddEnumerable(ServiceDescriptor.Singleton<IMySingletonService, MySingletonService>());
services.TryAddEnumerable(ServiceDescriptor.Singleton<IMySingletonService, MySingletonService>());
可以通过这种方式调用,如果注册信息相同就不注册,如果不同就注册。
这样就只有一个。
如果你只想注册一个的话可以这样:
services.AddSingleton<IMySingletonService, MySingletonService>();
services.TryAddSingleton<IMySingletonService, MySingletonService>();
services.TryAddSingleton<IMySingletonService, MySingletonService>();
这个就是如果有注册信息,就不再注册了。
还有其他的如替换注册信息,移除注册信息等,可以参考官网。
然后可以关注一下这个using Microsoft.Extensions.DependencyInjection.Extensions;,这里面是一些扩展的,如有自己需要的一般都在这里面了。
另一个值得注意的地方就是泛型模板的注册。
public interface IGeneticService<T>
{
public T getData();
}
public class GeneticService<T>: IGeneticService<T>
{
public T Data { get; private set; }
public GeneticService(T data)
{
this.Data = data;
}
public T getData()
{
return Data;
}
}
那么通过这种注册:
services.AddSingleton(typeof(IGeneticService<>),typeof(GeneticService<>));
然后实现一个接口:
[HttpGet]
public int GetService([FromServices] IMySingletonService mySingletonService, [FromServices] IGeneticService<IMySingletonService> geneticService)
{
Console.WriteLine(geneticService.getData().GetHashCode());
Console.WriteLine(mySingletonService.GetHashCode());
return 1;
}
结
这里埋了一个点:
services.AddSingleton<IMySingletonService, MySingletonService>();
services.AddSingleton<IMySingletonService>(new MySingletonService());
services.AddSingleton<IMySingletonService>(ServiceProvider =>
{
return new MySingletonService();
});
那就是上面这三种注册方式的对象释放行为是不一样的,如果不了解的话,那么可能你的服务跑着跑着内存就跑高了。下一节将会解释到。
上述只是个人整理,如有错误,望请指出,谢谢,一天一更。上述只是应用,如代码原理,请查看<<重新整理.net core 1400篇>>。
重新整理 .net core 实践篇————依赖注入应用[二]的更多相关文章
- 重新整理 .net core 实践篇————依赖注入应用之援军[四]
前言 介绍第三方依赖注入框架Autofac,看看为我们解决什么问题. 下面介绍4个点: 命名注册 属性注册 aop 注入 子容器命名 正文 为什么我们需要使用第三方框架?第三方框架为我们做了什么?第三 ...
- 重新整理 .net core 实践篇————依赖注入应用之生命法则[三]
前言 该章演示依赖注入中,对象的释放行为. 紧接上文表示,演示: services.AddSingleton<IMySingletonService, MySingletonService> ...
- 重新整理 .net core 实践篇—————HttpClientFactory[三十二]
前言 简单整理一下HttpClientFactory . 正文 这个HttpFactory 主要有下面的功能: 管理内部HttpMessageHandler 的生命周期,灵活应对资源问题和DNS刷新问 ...
- 重新整理 .net core 实践篇—————工作单元模式[二十六]
前言 简单整理一下工作单元模式. 正文 工作单元模式有3个特性,也算是其功能: 使用同一上下文 跟踪实体的状态 保障事务一致性 工作单元模式 主要关注事务,所以重点在事务上. 在共享层的基础建设类库中 ...
- 重新整理 .net core 实践篇—————路由和终结点[二十三]
前言 简单整理一下路由和终节点. 正文 路由方式主要有两种: 1.路由模板方式 2.RouteAttribute 方式 路由约束: 1.类型约束 2.范围约束 3.正则表达式 4.是否必选 5.自定义 ...
- 重新整理 .net core 实践篇————配置应用[一]
前言 本来想整理到<<重新整理.net core 计1400篇>>里面去,但是后来一想,整理 .net core 实践篇 是偏于实践,故而分开. 因为是重新整理,那么就从配置开 ...
- ASP.NET Core之依赖注入
本文翻译自:http://www.tutorialsteacher.com/core/dependency-injection-in-aspnet-core ASP.NET Core支持依赖注入,依赖 ...
- 【半小时大话.net依赖注入】(下)详解AutoFac+实战Mvc、Api以及.NET Core的依赖注入
系列目录 上|理论基础+实战控制台程序实现AutoFac注入 下|详解AutoFac+实战Mvc.Api以及.NET Core的依赖注入 前言 本来计划是五篇文章的,每章发个半小时随便翻翻就能懂,但是 ...
- 几十行代码实现ASP.NET Core自动依赖注入
在开发.NET Core web服务的时候,我们习惯使用自带的依赖注入容器来进行注入. 于是就会经常进行一个很频繁的的重复动作:定义一个接口->写实现类->注入 有时候会忘了写Add这一步 ...
随机推荐
- Spring框架的介绍
1.Spring框架的结构 由持久层.表现层.中间模块和测试层组成. 持久层:和数据接触.事务管理 表现层:对数据进行处理 中间模块:核心功能 测试层:用来测试完整度 核心功能介绍 1.1 控制反转 ...
- java例题_24 逆向输入数字
1 /*24 [程序 24 根据输入求输出] 2 题目:给一个不多于 5 位的正整数,要求:一.求它是几位数,二.逆序打印出各位数字. 3 */ 4 5 /*分析 6 * 首先从键盘得到一个正整数,不 ...
- 如何开发一个APP——转自知乎
作者:简单点链接:https://www.zhihu.com/question/22999185/answer/155469014来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注 ...
- [模拟]P1046 陶陶摘苹果
陶陶摘苹果 ## 题目描述 陶陶家的院子里有一棵苹果树,每到秋天树上就会结出10个苹果.苹果成熟的时候,陶陶就会跑去摘苹果.陶陶有个30厘米高的板凳,当她不能直接用手摘到苹果的时候,就会踩到板凳上再试 ...
- 自动化kolla-ansible部署ubuntu20.04+openstack-victoria之镜像制作centos6.5-14
自动化kolla-ansible部署ubuntu20.04+openstack-victoria之镜像制作centos6.5-14 欢迎加QQ群:1026880196 进行交流学习 制作OpenSta ...
- Spring学习笔记(五):JDBCTemplate+事务管理
1 概述 Spring为开发者提供了JDBCTemplate,可以简化很多数据库操作相关的代码,本文主要介绍JDBCTemplate的使用以及事务管理功能. 2 JDBC Template 2.1 配 ...
- 【Azure Developer】调用SDK的runPowerShellScript方法,在Azure VM中执行PowerShell脚本示例
当需要通过代码的方式执行PowerShell脚本时,可以参考以下的示例. Azure SDK中提供了两个方法来执行PowerShell脚本 (SDK Source Code: https://gith ...
- Mybatis3源码笔记(八)小窥MyBatis-plus
前言 Mybatis-Plus是一个 MyBatis增强工具包,简化 CRUD 操作,在 MyBatis 的基础上只做增强不做改变,为简化开发.提高效率而生,号称无侵入,现在开发中比较常用,包括我自己 ...
- Java文档注释全攻略
注释:注释起到对代码标注和解释的作用,如果你去看看JDK源码,会发现他们有许多的注释,而且注释是比代码还要多的,可见为代码添加注释是非常重要的,写好注释能让别人更加容易看懂你的代码,注释可以分为以下三 ...
- 关于Snowflake 生成53位ID
1, bug现象: 没有经过处理的Snowflake 生成的是64位bit的唯一ID,但由于多数时候我们前台用到js,但是js只支持53位bit的数值.这样就导致了传到前台的64位的丢失精度. 解决思 ...