杀手需要一把枪来执行刺杀计划,这里从杀手的角度来看,对枪存在一种依赖关系,没有枪就不能成功刺杀。代码中表
示如下:
public class Killer
{
Gun gun = new Gun();
}
杀手只是需要一把枪,至于枪是如何生产的,杀手并不关心,我们改写下代码如下:
复制代码
public class Killer
{
private Gun _gun;
public Killer(Gun gun)
{
_gun = gun;
}
}
复制代码
接着我们在Main方法中看下:
Killer的构造方法中需要Gun,我需要你。。。,即是我(Killer)对你(Gun)有所依赖,这就是依赖。
那什么是依赖注入呢?既然杀手需要一把枪,那么Main方法就提供一个Gun对象并赋予给Killer,代码如下:
复制代码
public class Program
{
static void Main(string[] args)
{
Gun gun = new Gun();
Killer killer = new Killer(gun);
}
}
复制代码
在我看来Main方法赋予给杀手gun对象的”过程“,即把你所需要的赋给你的”过程“这就是依赖注入。
了解了什么是依赖注入?那它有什么好处呢?亦或者说依赖注入的目的是什么?
在网上查找依赖注入概念时,控制反转往往与其同时出现。那么好,我们先来看看控制反转
三、控制反转
既然叫控制反转,那一定存在正向的了,那么什么算是正向的呢?
还是以杀手为例,
public Killer()
{
Gun gun = new Gun();//创造一把枪
}
杀手自己创造一把枪,也就是自动去new即是正向。
知道了正向,那么反向就不难理解了,杀手本身不去new,被动获取即是反向。那何来控制一说呢?
好我们继续看,先正向看,杀手创造一把枪(获取枪消耗时间,存在风险,可能错过最佳行刺时间)然后去刺杀,刺杀
完成后还要销毁枪(销毁枪消耗时间,这个时间可能被抓,风险太大)。
复制代码
public Killer()
{
Gun gun = new Gun();//创造一把枪
gun.Kill();//去行刺
gun.Dispose();//销毁枪
}
复制代码
既然创造枪和销毁枪都有风险,和不将这两部分交给其他人去做?下面反向来了
杀手在整个刺杀行动中,首先会有专人给杀手枪(这个过程就是依赖注入),杀手完成刺杀行动后,会有专人进行枪的
善后处理。
即把枪除了刺杀之外的整个枪的使用过程的控制权完全交给第三方。
总结一下:这里的使用过程即可理解为对象(枪)的生命周期,第三方即是容器。容器接管了对象的创建、销毁的控制
权。容器通过查找相关依赖动态的将对象(枪)注入给需求方(杀手)。
这种自身并不负责依赖对象的创建及销毁。由容器来管理控制的思想称之为控制反转。
这里可以回答上面遗留的问题,依赖注入的目的是什么?目的是实现控制反转。
四、.net core中使用依赖注入
了解了依赖注入和控制反转后,来看看.net core中在哪里进行的依赖注入,接下来回归我们的项目
1、.net core自身的IOC容器
NET Core自身集成了一个轻量级的IOC容器,只要在Startup.cs的ConfigureServices方法中进行配置即可,如之前配置
的用户信息服务
public void ConfigureServices(IServiceCollection services)
{
services.AddScoped<IUserInfoRepository, UserInfoRepository>();
services.AddScoped<IUserInfoServices, UserInfoServices>();
}
这里需要注意一点,配置注入的生命周期。有三种注入的生命周期
AddSingleton 单一实例对象整个程序运行期间都是相同的(相当于单例模式,例如:全局配置信息,统计在线人数)
AddScoped 对每次请求而言对象是相同的,但在请求之间不同(例如:获取用户信息,每个用户均请求自身的对象)
AddTransient 每次请求对象都是不同的(暂未找到合适的应用场景,如有合适的请告知)
.Net Core自身的IOC容器比较简单,如果想要更多的功能和扩展,还需要第三方的框架支持。
2、第三方IOC容器
.Net Core 有多种三方容器 如:Autofac、DryIoc、Grace、lightInject等等。评价较好的是Autofac,我们这里也使
用Autofac
在 Package Manager Console中输入如下命令,安装Autofac包
Install-Package Autofac -Version 6.1.0
Install-Package Autofac.Extensions.DependencyInjection -Version 7.1.0
替换默认的IOC容器,在Program类中的CreateHostBuilder方法中 将默认ServiceProviderFactory指定为
AutofacServiceProviderFactory
复制代码
public static IHostBuilder CreateHostBuilder(string[] args) =>
Host.CreateDefaultBuilder(args)
.UseServiceProviderFactory(new AutofacServiceProviderFactory())
.ConfigureWebHostDefaults(webBuilder =>
{
webBuilder.UseStartup<Startup>();
});
复制代码
在Startup类中添加如下方法:
复制代码
public void ConfigureContainer(ContainerBuilder builder)
{
var basePath = AppContext.BaseDirectory;
//Service所在程序集
string servicePath = Path.Combine(basePath, "MServices.dll");
Assembly serviceDll = Assembly.LoadFrom(servicePath);
//Repository层所在程序集
string repositoryPath = Path.Combine(basePath, "MRepository.dll");
Assembly repositoryDll = Assembly.LoadFrom(repositoryPath);
builder.RegisterAssemblyTypes(serviceDll)
.AsImplementedInterfaces()
.InstancePerDependency();
builder.RegisterAssemblyTypes(repositoryDll)
.AsImplementedInterfaces()
.InstancePerDependency();
}
复制代码
移除之前ConfigureServices中的注入
services.AddScoped<IUserInfoRepository, UserInfoRepository>();
services.AddScoped<IUserInfoServices, UserInfoServices>();
运行项目,并调用
亚马逊测评 www.yisuping.cn