Microsoft.Extensions.DependencyInjection中的Transient依赖注入关系,使用不当会造成内存泄漏
Microsoft.Extensions.DependencyInjection中(下面简称DI)的Transient依赖注入关系,表示每次DI获取一个全新的注入对象。但是使用Transient依赖注入关系时,最好要配合IServiceScope来一起使用,因为通过Transient依赖注入关系创建的对象,都会被创建它的ServiceProvider对象内部引用,这样会造成注入对象无法被GC及时回收,造成内存泄漏,只有当调用ServiceProvider对象的Dispose方法后,ServiceProvider才会解除其内部对注入对象的引用,之后这些注入对象才能被GC回收。
我们新建一个.NET Core控制台项目,然后假设我们有接口IPeople和实现类People,他们之间的依赖注入关系是Transient。
现在,如果我们的代码中有一个for循环,它会循环1000次,每一次都会从DI中获取一个IPeople对象实例,由于接口IPeople和类People是Transient关系,所以每次DI都会创建一个新的People对象实例。但是我们只需要在每次循环中,调用People类的DoSomething方法做一些事情后,就不需要创建的People对象了,也就是说我们希望每次循环结束后,GC都能尽量回收在循环中创建的People对象实例。
所以我们下了下面的代码:
using Microsoft.Extensions.DependencyInjection;
using System; namespace NetCoreDITransientInScope
{
interface IPeople
{
void DoSomething();
} class People : IPeople
{
public void DoSomething()
{
Console.WriteLine("DoSomething is running");
}
} class Program
{ static void Main(string[] args)
{
IServiceCollection services = new ServiceCollection();
services.AddTransient<IPeople, People>();//注册接口IPeople和类People的关系为Transient using (ServiceProvider rootServiceProvider = services.BuildServiceProvider())
{
//执行1000次循环,每一次循环创建一个People对象实例,在rootServiceProvider调用Dispose方法前,创建的1000个People对象实例都不会被GC回收
for (int i = 0; i < 1000; i++)
{
IPeople people = rootServiceProvider.GetService<IPeople>();
people.DoSomething(); //在每次循环结束后,创建的People对象实例无法被GC回收,因为在rootServiceProvider的内部对所有创建的Transient对象都保持了引用,除非调用rootServiceProvider的Dispose方法,否则在每次循环中创建的People对象实例都无法被GC回收
}
} Console.WriteLine("Press any key to end...");
Console.ReadKey();
}
}
}
从上面代码的注释中,我们可以看到,实际上每一次for循环执行完后,GC并不能立即回收在循环中创建的People对象实例,原因是ServiceProvider对象rootServiceProvider的内部引用了由DI创建的所有People对象实例,除非调用rootServiceProvider的Dispose方法(也就是在上面using代码块最后),否则所有的People对象实例都无法被GC回收。设想一下,如果将上面的for循环改为一个死循环(对于有些后台服务程序而言,的确需要死循环),那么DI会创建大量的People对象实例无法被GC及时回收,造成内存泄漏。
所以正确使用Transient依赖注入关系的方法应该是,配合IServiceScope对象来使用,我们将上面的代码改为如下:
using Microsoft.Extensions.DependencyInjection;
using System; namespace NetCoreDITransientInScope
{
interface IPeople
{
void DoSomething();
} class People : IPeople
{
public void DoSomething()
{
Console.WriteLine("DoSomething is running");
}
} class Program
{ static void Main(string[] args)
{
IServiceCollection services = new ServiceCollection();
services.AddTransient<IPeople, People>();//注册接口IPeople和类People的关系为Transient using (ServiceProvider rootServiceProvider = services.BuildServiceProvider())
{
//执行1000次循环,每一次循环创建一个People对象实例
for (int i = 0; i < 1000; i++)
{
//在每一次循环中创建一个IServiceScope对象serviceScope,然后使用serviceScope的ServiceProvider创建People对象实例,这样在rootServiceProvider中并没有对People对象实例的引用,只有每次循环中创建的serviceScope中的ServiceProvider保持了People对象实例的引用,这样在每次循环中当调用serviceScope的Dispose方法后,单次循环中创建的People对象实例就可以被GC回收了,而不是等到rootServiceProvider调用Dispose方法后,才能被GC回收
using (IServiceScope serviceScope = rootServiceProvider.CreateScope())
{
IPeople people = serviceScope.ServiceProvider.GetService<IPeople>();
people.DoSomething();
}
}
} Console.WriteLine("Press any key to end...");
Console.ReadKey();
}
}
}
从上面代码中,我们可以看到,由于现在在每次for循环中,是由一个独立的IServiceScope对象serviceScope的ServiceProvider,来创建People对象实例,所以在for循环外面的ServiceProvider对象rootServiceProvider,其并没有内部引用由DI创建的People对象实例。而在每次for循环中,我们都调用了serviceScope的Dispose方法(也就是在上面第二个using代码块最后),这样每次循环结束后,就没有任何代码引用循环内的People对象实例了,GC就可以及时回收由DI创建的People对象实例。
在使用Microsoft.Extensions.DependencyInjection的Transient依赖注入关系时,一定要注意本文所述的内存泄漏问题,这个问题可能很多才开始接触Microsoft.Extensions.DependencyInjection的开发人员不会注意到,但是它会严重影响你的程序性能和稳定性。可以参考下面两篇帖子中发帖人提出的问题:
IServiceProvider garbage collection / disposal
When are .NET Core dependency injected instances disposed?
也可以参考在GitHub上,微软官方对这个问题的讨论:
Revisit tracking transient services for disposal
Microsoft.Extensions.DependencyInjection中的Transient依赖注入关系,使用不当会造成内存泄漏的更多相关文章
- Microsoft.Extensions.DependencyInjection 之三:展开测试
目录 前文回顾 IServiceCallSite CallSiteFactory ServiceProviderEngine CompiledServiceProviderEngine Dynamic ...
- Microsoft.Extensions.DependencyInjection 之三:反射可以一战(附源代码)
目录 前文回顾 IServiceCallSite CallSiteFactory ServiceProviderEngine CompiledServiceProviderEngine Dynamic ...
- 使用 Microsoft.Extensions.DependencyInjection 进行依赖注入
没有 Autofac DryIoc Grace LightInject Lamar Stashbox Unity Ninject 的日子,才是好日子~~~~~~~~~~ Using .NET Core ...
- 使用诊断工具观察 Microsoft.Extensions.DependencyInjection 2.x 版本的内存占用
目录 准备工作 大量接口与实现类的生成 elasticsearch+kibana+apm asp.net core 应用 请求与快照 Kibana 上的请求记录 请求耗时的分析 请求内存的分析 第2次 ...
- Microsoft.Extensions.DependencyInjection 之二:使用诊断工具观察内存占用
目录 准备工作 大量接口与实现类的生成 elasticsearch+kibana+apm asp.net core 应用 请求与快照 Kibana 上的请求记录 请求耗时的分析 请求内存的分析 第2次 ...
- 【17MKH】我在框架中对.Net依赖注入的扩展
说明 依赖注入(DI)是控制反转(IoC)的一种技术实现,它应该算是.Net中最核心,也是最基本的一个功能.但是官方只是实现了基本的功能和扩展方法,而我呢,在自己的框架 https://github. ...
- DotNetCore跨平台~一起聊聊Microsoft.Extensions.DependencyInjection
写这篇文章的心情:激动 Microsoft.Extensions.DependencyInjection在github上同样是开源的,它在dotnetcore里被广泛的使用,比起之前的autofac, ...
- 解析 Microsoft.Extensions.DependencyInjection 2.x 版本实现
项目使用了 Microsoft.Extensions.DependencyInjection 2.x 版本,遇到第2次请求时非常高的内存占用情况,于是作了调查,本文对 3.0 版本仍然适用. 先说结论 ...
- Microsoft.Extensions.DependencyInjection 之一:解析实现
[TOC] 前言 项目使用了 Microsoft.Extensions.DependencyInjection 2.x 版本,遇到第2次请求时非常高的内存占用情况,于是作了调查,本文对 3.0 版本仍 ...
随机推荐
- CentOS6.10下安装MongoDB和Redis
安装mongodb 首先考虑离线安装,但是安装过程中在启动服务的时候出现了问题,centOS出于稳定原因考虑,系统自带的glibc版本过低, 而编译需要使用较高版本,这个问题我查询了一下,需要升级gl ...
- python循环输出
python 目录 python 1.九九乘法表 2.循环输出数字0-9,数字为六,跳出循环,执行其他循环,数字为八,结束循环 3.使用循环计算0-100素数的和 4.使用for循环输出三角形 1.九 ...
- golang map学习
当对map只声明时,由于map为引用类型,所以默认值为nil,但对nil map 而言,支持read ,但不支持write 当执行write操作时, 会抛出panic异常; 代码如下: func Te ...
- Do not use built-in or reserved HTML elements as component id:mask vue报错
今天学习了一下vue的组件,但是报了一个错误 Do not use built-in or reserved HTML elements as component id:mask , 经过查询得知是因 ...
- 容器云平台No.6~企业级分布式存储Ceph
简介 ceph作为一个统一的分布式存储系统,提供了高性能,高可用性,高扩展性.ceph的统一体现在其可以提供文件系统.块存储.对象存储,在云环境中,通常采用ceph作为后端存储来保证数据的高可用性. ...
- tensorflow-GPU配置
在使用GPU版的TensorFlow跑程序的时候,如果不特殊写代码注明,程序默认是占用所有主机上的GPU,但计算过程中只会用其中一块.也就是你看着所有GPU都被占用了,以为是在GPU并行计算,但实际上 ...
- MySql-8.0.x免安装版下载与配置,Navicat打开数据库链接报错1251的解决办法
若你以前卸载过mysql,小白极大可能没有卸载和删除干净残留,没有卸载干净就肯定重装不成功,可参考https://www.cnblogs.com/Luoters/p/11869032.html 参考与 ...
- SQL Server 子查询遇到的坑
这两天改 Bug 时使用 Sql Server 的子查询遇到了一些问题,特此记录一下,之前用 MySQL 比较多,按照 MySQL 的语法其实是没有问题的. 以下面这张表为例: 执行以下 SQL: s ...
- pip安装更换镜像源
说明 有时候网不好,pip安装非常慢,所以需要更换源,特记录如下 国内镜像地址: # 清华大学 https://pypi.tuna.tsinghua.edu.cn/simple # 豆瓣 http ...
- k8s下的jenkins如何设置maven
关于k8s环境的jenkins集群 k8s下搭建了jenkins集群后,执行任务时会新建pod,任务完成后pod被销毁,架构如下图所示: 在k8s搭建jenkins集群的步骤请参照<> 关 ...