EFCore:多线程中使用仓储方法,报错A second operation was started on this context instance before a previous operat

报错信息:上下文实例在销毁后又被第二次使用

System.InvalidOperationException: A second operation was started on this context instance before a previous operation completed. This is usually caused by different threads concurrently using the same instance of DbContext.

相关的代码示例:

resList.ForEach(async item =>
    {
        var patientInfo = await _patientInfoManager.GetByPatientIndexAsync(item);
    });

原因:

DbContext生命周期默认注入是Scope,每一次请求时创建一个实例,在当前请求的上下文中共用,当请求结束后,释放生命周期,释放数据库链接。若开启多线程,在不同的线程中使用同一个DbContext上下文,则报错

相关内容

1. 使用async关键字标识的方法会在一个新的线程中执行。当调用这个方法时,它会在后台启动一个新的任务,并在完成后返回结果。这使得我们可以在调用异步方法的同时继续执行其他任务,而不必等待异步方法完成。

解决方案:

1. forEach 改为正常 for 循环

2. 尝试使用 Parallel.ForEachAsync (为验证)

3. 在构造函数注入IServiceProvider,应用中通过IServiceProvider的GetService方法获取实例

 https://www.cnblogs.com/zxsn2014/p/16478922.htmlhttps://www.cnblogs.com/zxsn2014/p/16478922.html

4. 关于3的优化解决方案

abpvnext 开发中ValidationErrors和LifetimeScope异常的解决办法_abp vnext 异常处理_吹牛不交税的博客-CSDN博客abpvnext Method arguments are not valid! See ValidationErrors for detailsInstances cannot be resolved and nested lifetimes cannot be created from this LifetimeScope as it (or one of its parent scopes) has already been disposed._abp vnext 异常处理https://blog.csdn.net/yangyong1250/article/details/129752279

2023/8/17

改为ETO通知,继承自ISingletonDependency 单例模式

这种方式感觉可以解决当前的问题

因为我在application层使用了多线程去添加定时任务,使用消息通知将每一个消费者都作为单例模式去消费,就不会出现DbContent上下文销毁的问题。

2023/8/17  个人的解决方案

application层使用 Paella多线程去去处理这个ListA, 每一个线程中不使用仓储方法,而是触发一个Eto出去,在Eto的handle方法内去使用仓储方法。

注意handle要继承自ISingletonDependency,实现单例模式,保证了每个DbContent都是单例,线程之间互相不影响。

2023/8/18 亲测有效的方法

代码中相干的代码段直接开启一个工作单元 

使用IUnitOfWorkManager.Begin()方法,如下所示:

public class MyService
{
    private readonly IUnitOfWorkManager _unitOfWorkManager;
    private readonly IPersonRepository _personRepository;
    private readonly IStatisticsRepository _statisticsRepository;

    public MyService(IUnitOfWorkManager unitOfWorkManager, IPersonRepository personRepository, IStatisticsRepository statisticsRepository)
    {
        _unitOfWorkManager = unitOfWorkManager;
        _personRepository = personRepository;
        _statisticsRepository = statisticsRepository;
    }

    public void CreatePerson(CreatePersonInput input)
    {
        var person = new Person { Name = input.Name, EmailAddress = input.EmailAddress };

        using (var unitOfWork = _unitOfWorkManager.Begin())
        {
            _personRepository.Insert(person);
            _statisticsRepository.IncrementPeopleCount();

            unitOfWork.Complete();
        }
    }
}

Reference:

1. https://www.cnblogs.com/zxsn2014/p/16478922.html

2. abpvnext 开发中ValidationErrors和LifetimeScope异常的解决办法_abp vnext 异常处理_吹牛不交税的博客-CSDN博客

  • 5
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

董厂长

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值