EF异常‘在创建模型,此时不可使用上下文。如果在 OnModelCreating 方法内使用上下文或如果多个线程同时访问同一上下文实例,可能引发此异常。 请注意不保证 DbContext 的实例成员’

 一、EF的线程安全问题

1.在使用EF是经常会抛出这样的异常

其他信息: 正在创建模型,此时不可使用上下文。如果在 OnModelCreating 方法内使用上下文或如果多个线程同时访问同一上下文实例,可能引发此异常。
请注意不保证 DbContext 的实例成员和相关类是线程安全的。


2.说明

1.在使用EF时很多时候不能对上线文进行单例处理,也不能将上线文设置为static类型

2.如果对于使用上线文对象不太注意就会出现,多线程的如上异常。

3.官方给出的上线文的线程安全说明:此类型的任何公共 static成员都是线程安全的。但不保证所有实例成员都是线程安全的。

二、解决方案:

1.在写代码时特别注意:同一个上下文实例,不在多个线程中使用

2.在有等待代码操作,或定时器,或夹杂其他长时间执行代码的循环语句中注意:在这样特殊执行操作代码中,每次使用上下文都重新创建对象

3.在特殊循环操作中,重新创建对象,对于查询可以及时获取数据库更新到的数据。

举例说明一:同一个上下文实例,不在多个线程中使用

1.抛出异常的代码段

MenuOperate _menu = new MenuOperate();
ModuleOperate _module = new ModuleOperate();
Action update1 = () =>
{
    _menu.UpdateFirstName(Count.ToString());
    _module.UpdateFirstName(Count.ToString());

    Console.WriteLine(_menu.GetFirstName());
    Console.WriteLine(_module.GetFirstName());
    Count++;
};
for (int i = 0; i < 3; i++)
{
    Task.Factory.StartNew(update1); //启动线程访问数据库操作
}
2.解决方案的代码段

Action<object> update1 = (number) =>
{
    while (true)
    {
        //将上线文实例放在本线程中创建
        MenuOperate _menu = new MenuOperate();
        ModuleOperate _module = new ModuleOperate();
        _menu.UpdateFirstName(Count.ToString());
        _module.UpdateFirstName(Count.ToString());

        Console.WriteLine("-------");
        Console.WriteLine(_menu.GetFirstName());
        Console.WriteLine(_module.GetFirstName());

        Count++;
        Thread.Sleep(1000 * Convert.ToInt32(number));
    }
};
for (int i = 0; i < 3; i++)
{
    Task.Factory.StartNew(update1, i + 1);
}

举例说明二:在这样特殊执行操作代码中,每次使用上下文都重新创建对象

特别说明:此处在每一次循环中都重现创建上下文

//显示菜单
Task.Factory.StartNew(() =>
{
    while (true)
    {
        MenuOperate _menu = new MenuOperate();
        string menuname = _menu.GetFirstName();
        Console.WriteLine($"---时间:{DateTime.Now.ToLongTimeString()},菜单名称:{menuname}");
        Thread.Sleep(1000);
    }
});
//显示模块
Task.Factory.StartNew(() =>
{
    while (true)
    {
        ModuleOperate _menu = new ModuleOperate();
        string name = _menu.GetFirstName();
        Console.WriteLine($"***时间:{DateTime.Now.ToLongTimeString()},模块名称:{name}");
        Thread.Sleep(1000);
    }
});

更多:

EntiryFramework中事务操作(三)事务回滚数据模型和数据库不对应问题

EF,‘正在创建模型,此时不可使用上下文。如果在 OnModelCreating 方法内使用上下文或如果多个线

EF中Sum()异常:到值类型“System.Decimal”的强制转换失败,因为具体化值为 null。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值