ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

重新整理 .net core 实践篇————依赖注入应用之生命法则[三]

2021-05-27 23:32:49  阅读:212  来源: 互联网

标签:容器 AddSingleton core stop 重新整理 hostApplicationLifetime services net public


前言

该章演示依赖注入中,对象的释放行为。

紧接上文表示,演示:

services.AddSingleton<IMySingletonService, MySingletonService>();
services.AddSingleton<IMySingletonService>(new MySingletonService());
services.AddSingleton<IMySingletonService>(ServiceProvider =>
{
	return new MySingletonService();
});

这三种生命周期的注入方式的不同,他们的生命周期不同。

主要是两条法则:

  1. DI 只负责释放由其创建的对象实例,DI 在容器或者子容器释放时,释放由其创建的对象的实例。

  2. 不要在根容器中获取瞬时服务,不要在根容器中获取瞬时服务,不要在根容器中获取瞬时服务,重要的事情说三遍。

正文

第一条法则证明公理:

public interface ITestService
{

}

public class TestService:ITestService,IDisposable
{
	public void Dispose()
	{
		Console.WriteLine($"DisposableTestService Disposed:{this.GetHashCode()}");
	}
}

注册:

services.AddSingleton<ITestService>(new TestService());

然后测试:

[HttpGet]
public int GetService([FromServices] ITestService testService,[FromServices]IHostApplicationLifetime hostApplicationLifetime,[FromQuery]bool stop = false)
{
	if (stop)
	{
		hostApplicationLifetime.StopApplication();
	}
	return 1;
}

结果:

应用离开的时候并没有调用我们的Dispose方式。

那么换一种注册方式:

services.AddSingleton<ITestService, TestService>();

结果如下:

应用程序离开的时候会调用我们的dispose。

再换一种方式:

services.AddSingleton<ITestService>(serviceProvider =>
{
     return new TestService();
});

也会主动调用这个dispose。

证明不要在根容器中获取瞬时服务:

理论上离开区域那么应该会调用dispose 释放,但是情况和我们想的不一样。

测试代码:

[HttpGet]
public int GetService([FromServices]IHostApplicationLifetime hostApplicationLifetime,[FromQuery]bool stop = false)
{
	if (stop)
	{
		hostApplicationLifetime.StopApplication();
	}
	return 1;
}

当访问stop参数不为true


当访问stop参数为true,也就是关闭服务的时候:

说明根容器获取的瞬时对象,并没有在离开区域内释放,而是在容器回收的时候才释放。间接的说明一个问题,这个对象起码被我们的根容器间接引用了,这里埋个坑,后面系列解释为什么。

那么为什么我们从FromServices中获取的ITestService会被释放呢?

这是应为,其实每一次http访问都会创建一个子容器来保证隔离:

[HttpGet]
public int GetService([FromServices]IHostApplicationLifetime hostApplicationLifetime,[FromQuery]bool stop = false)
{
	using (IServiceScope scope = HttpContext.RequestServices.CreateScope())
	{
		var obj=scope.ServiceProvider.GetService<ITestService>();
	}
	if (stop)
	{
		hostApplicationLifetime.StopApplication();
	}
	return 1;
}

结果:

下一节介绍Autofac,主要介绍一下什么情况我们需要使用第三方的Autofac,它给我们带来什么东西,或者给我们带来什么方便,以及如何操作。Autofac 还是比较重要的,一般大一点的项目都会用上,比较方便吧。

上述只是个人整理,如果错误,望请指教。

标签:容器,AddSingleton,core,stop,重新整理,hostApplicationLifetime,services,net,public
来源: https://www.cnblogs.com/aoximin/p/14809538.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有