dotnet core在Task中使用依赖注入的Service/EFContext

时间:2023-01-19 14:42:21

C#:在Task中使用依赖注入的Service/EFContext

dotnet core时代,依赖注入基本已经成为标配了,这就不多说了.

前几天在做某个功能的时候遇到在Task中使用EF DbContext的问题,学艺不精的我被困扰了不短的一段时间,

于是有了这个文章.

先说一下代码结构和场景.

首先有一个HouseDbContext,代码大概是下面这样:

public class HouseDbContext : DbContext
{
public HouseDbContext(DbContextOptions<HouseDbContext> options)
: base(options)
{
}
public DbSet<Notice> Notices { get; set; }
}

接着已经在StarUp.cs中初始化并注入了,注入代码是这样的:


services.AddDbContextPool<HouseDbContext>(options =>
{
options.UseLoggerFactory(loggerFactory);
options.UseMySql(Configuration["MySQLString"].ToString());
});

有一个NoticeService.cs 会用到HouseDbContext 进行增删查改

public class NoticeService
{ private readonly HouseDbContext _dataContext; public NoticeService(HouseDbContext dataContext)
{
_dataContext = dataContext;
} public Notice FindNotice(long id)
{
var notice = _dataContext.Notices.FirstOrDefault(n =>n.Id == id);
return notice;
}
}

当然我们也需要把NoticeService注入到容器中,类似这样


services.AddScoped<NoticeService,NoticeService>();

现在一切都是很美好的,也能正常查询出Notice

然后某一天来了,有个需求是Update Notice之后需要把Notice同步到另外一个地方,例如Elasticsearch?

代码如下:


public class NoticeService
{ private readonly HouseDbContext _dataContext; public NoticeService(HouseDbContext dataContext)
{
_dataContext = dataContext;
} public Notice FindNotice(long id)
{
var notice = _dataContext.Notices.FirstOrDefault(n =>n.Id == id);
return notice;
} public void Save(Notice notice)
{
_dataContext.Notices.Add(notice);
_dataContext.SaveChanges();
Task.Run(() =>
{
try
{
var one = _dataContext.Notices.FirstOrDefault(n =>n.Id == notice.Id);
// write to other
}
catch (Exception ex)
{
Console.WriteLine(ex.ToString());
}
});
}
}

然后一跑...

代码炸了...

恭喜你获得跨线程使用EF DbContext导致上下文不同步的异常.

错误大概长这样.


System.ObjectDisposedException: Cannot access a disposed object. A common cause of this error is disposing a context that was resolved from dependency injection and then later trying to use the same context instance elsewhere in your application. This may occur if you are calling Dispose() on the context, or wrapping the context in a using statement. If you are using dependency injection, you should let the dependency injection container take care of disposing context instances.
Object name: 'HouseDbContext'.

估计现在整个人都不好了.

这个撒意思呢?


无法访问被释放的对象。 这种错误的一个常见原因是使用从依赖注入中解决的上下文,然后在应用程序的其他地方尝试使用相同的上下文实例。 如果您在上下文上调用Dispose(),或者在using语句中包装上下文,可能会发生这种情况。如果使用依赖项注入,则应该让依赖项注入容器处理上下文实例。

用人话来说是什么意思呢?

这里的HouseDbContext是依赖注入进来的,生命周期由容器本身管理;

在Task.Run中再次使用HouseDbContext实例中由于已经切换了线程了,

HouseDbContext实例已经被释放掉了,无法再继续使用同一个实例,我们应该自己初始化HouseDbContext来用.

到这里的话,上次我做的时候心生一计:

既然我们不能直接从构造函数注入的HouseDbContext实例的话,我们是不是可以直接从依赖注入容器中拿一个实例回来呢?

那在dotnet core里面可以用个什么从容器中取出实例呢?

答案是:IServiceProvider

代码如下:


public class NoticeService
{ private readonly HouseDbContext _dataContext; private readonly IServiceProvider _serviceProvider; public NoticeService(HouseDbContext dataContext,IServiceProvider serviceProvider)
{
_dataContext = dataContext;
_serviceProvider = serviceProvider;
} public Notice FindNotice(long id)
{
var notice = _dataContext.Notices.FirstOrDefault(n => n.Id == id);
Task.Run(() =>
{
try
{
var context = _serviceProvider.GetService<HouseDbContext>();
var one = context.Notices.FirstOrDefault(n => n.Id == notice.Id);
Console.WriteLine(notice.Id);
// write to other
}
catch (Exception ex)
{
Console.WriteLine(ex.ToString());
}
});
return notice;
} public void Save(Notice notice)
{
_dataContext.Notices.Add(notice);
_dataContext.SaveChanges();
Task.Run(() =>
{
try
{
var one = _dataContext.Notices.FirstOrDefault(n => n.Id == notice.Id);
// write to other
}
catch (Exception ex)
{
Console.WriteLine(ex.ToString());
}
});
}
}

跑一下看看...

然而事实告诉我,实例是能拿得到,然而还是会炸,错误是一样的.

原因其实还是一样的,这里已经不受依赖注入托管了,人家的上下文你别想用了.

那咋办呢...

在EF6,还可以直接new HouseDbContext 一个字符串进去初始化,在EF Core这里,已经不能这样玩了.

那可咋办呢?

翻了好多资料都没看到有人介绍过咋办,最后居然还是在官网教程里面找到了样例.

先看代码...

 Task.Run(() =>
{
try
{
var optionsBuilder = new DbContextOptionsBuilder<HouseDbContext>();
// appConfiguration.MySQLString appConfiguration是配置类,MySQLString为连接字符串
optionsBuilder.UseMySql(appConfiguration.MySQLString);
using (var context = new HouseDbContext(optionsBuilder.Options))
{
var one = context.Notices.FirstOrDefault(n => n.Id == notice.Id);
// 当然你也可以直接初始化其他的Service
var nService = new NoticeService(context,null);
var one =nService.FindOne(notice.Id);
} }catch (Exception ex)
{
Console.WriteLine(ex.ToString());
}
});

教程代码在:Configuring a DbContext

大功告成...

dotnet core在Task中使用依赖注入的Service/EFContext的更多相关文章

  1. 在&period;NET Core控制台程序中使用依赖注入

    之前都是在ASP.NET Core中使用依赖注入(Dependency Injection),昨天遇到一个场景需要在.NET Core控制台程序中使用依赖注入,由于对.NET Core中的依赖注入机制 ...

  2. ASP&period;NET Core - 在ActionFilter中使用依赖注入

    上次ActionFilter引发的一个EF异常,本质上是对Core版本的ActionFilter的知识掌握不够牢固造成的,所以花了点时间仔细阅读了微软的官方文档.发现除了IActionFilter.I ...

  3. ASP&period;NET Core 新建线程中使用依赖注入的问题

    问题来自博问的一个提问 .net core 多线程数据保存的时候DbContext被释放 . TCPService 通过构造函数注入了 ContentService , ContentService ...

  4. 如何在&period;NET Core控制台程序中使用依赖注入

    背景介绍 依赖注入(Dependency Injection), 是面向对象编程中的一种设计原则,可以用来减低代码之间的耦合度.在.NET Core MVC中 我们可以在Startup.cs文件的Co ...

  5. ASP&period;NET Core中的依赖注入(1):控制反转(IoC)

    ASP.NET Core在启动以及后续针对每个请求的处理过程中的各个环节都需要相应的组件提供相应的服务,为了方便对这些组件进行定制,ASP.NET通过定义接口的方式对它们进行了"标准化&qu ...

  6. ASP&period;NET Core中的依赖注入(2):依赖注入(DI)

    IoC主要体现了这样一种设计思想:通过将一组通用流程的控制从应用转移到框架之中以实现对流程的复用,同时采用"好莱坞原则"是应用程序以被动的方式实现对流程的定制.我们可以采用若干设计 ...

  7. 深入理解net core中的依赖注入、Singleton、Scoped、Transient(三)

    相关文章: 深入理解net core中的依赖注入.Singleton.Scoped.Transient(一) 深入理解net core中的依赖注入.Singleton.Scoped.Transient ...

  8. 深入理解net core中的依赖注入、Singleton、Scoped、Transient(二)

    相关文章: 深入理解net core中的依赖注入.Singleton.Scoped.Transient(一) 深入理解net core中的依赖注入.Singleton.Scoped.Transient ...

  9. ASP&period;NET Core 中文文档 第四章 MVC(3&period;8)视图中的依赖注入

    原文:Dependency injection into views 作者:Steve Smith 翻译:姚阿勇(Dr.Yao) 校对:孟帅洋(书缘) ASP.NET Core 支持在视图中使用 依赖 ...

随机推荐

  1. 1122Shell脚本之利用mysqldump备份MySQL数据库

    #!/bin/bash #Mysql 自动备份 压缩并上传到 指定ftp #设想每天凌晨3点备份mysql #编辑crontab配置文件 #00 03 * * * backupmysql.sh #压缩 ...

  2. 使用jsPlumb制作流程图设计器

    jsPlumb是一个比较强大的绘图组件,它提供了一种方法,主要用于连接网页上的元素.在现代浏览器中,它使用SVG或者Canvas技术,而对于IE8以下(含IE8)的古董浏览器,则使用VML技术. 项目 ...

  3. 关于BigDecimal的使用

    为什么使用BigDecimal 使用BigDecimal首先要注意到float,double是无法支持商业计算的.只能支持工程计算.即误差允许的计算.通常float占用4个字节,32位.double占 ...

  4. Foundation Data Structure

    LinkedList : /** * 考虑的比较的周全,并且包含了全部的情况,代码也不乱<b></b> * * @param index * 插入的位置 * @param c ...

  5. mockServer学习

    mockServer学习 很喜欢mockserver官方主页的背景颜色和格式 官方主页如下: http://www.mock-server.com/

  6. 无废话MVC入门教程二&lbrack;第一个小Demo&rsqb;

    mvc技术交流,欢迎加群: 本文目标 1.了解"模型"."视图"."控制器"的创建.调试和使用过程. 本文目录 1.创建模型 2.创建视图 ...

  7. &lpar;C&num;&rpar;Windows Shell 外壳编程系列8 - 同后缀名不同图标?

    原文 (C#)Windows Shell 外壳编程系列8 - 同后缀名不同图标? (本系列文章由柠檬的(lc_mtt)原创,转载请注明出处,谢谢-) 接上一节:(C#)Windows Shell 外壳 ...

  8. 图论——Dijkstra算法

    图论其实是比较难的一种题型,但是一些模板题,是没有什么太大难度的! 这里给大家带来的是迪杰斯特拉(Dijkstra)算法. 迪杰斯特拉算法是由荷兰计算机科学家狄克斯特拉于1959 年提出的,因此又叫狄 ...

  9. EasyPR源码剖析(7):车牌判断之SVM

    前面的文章中我们主要介绍了车牌定位的相关技术,但是定位出来的相关区域可能并非是真实的车牌区域,EasyPR通过SVM支持向量机,一种机器学习算法来判定截取的图块是否是真的“车牌”,本节主要对相关的技术 ...

  10. Nginx&plus;uwsgi部署 Diango(生产环境)

    环境:CentOS6.5 + Nginx1.11.5 + Python3.5.2 1. 安装基础软件包 yum install -y zlib-devel bzip2-devel \ pcre-dev ...