code first提示已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭解决方法

Posted 奈何缘浅丶世俗沾染了无奈

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了code first提示已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭解决方法相关的知识,希望对你有一定的参考价值。

使用codefirst查询当然是必不可少的,但有时不小心可能很简单的查询也会导致异常。

下面用codefirst做个示例简单演示下异常发生的场景:

var attendlist = db.Database.SqlQuery<AttendCountViewModel>("select STUDENTID,SUM(isnull(REALHOUR,0)) As REALHOUR from ATTEND_LIST where ATTENDDATE=‘" + geDate + "‘ and GOTIME is not null and GETOFFTIME is not null group by STUDENTID");
 
            foreach (AttendCountViewModel item in attendlist)
            {
                STUDENT_INFO student = db.STUDENT_INFO.Find(item.STUDENTID);
                db.SaveChanges();
            }

逻辑比较简单,就是首先查询一个AttendCountViewModel列表,并当作IEnumerable<AttendCountViewModel>对象进行读取的时候,在里面再进行查询操作,不出意外的话,程序会毫不留情的报错:

已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭。                                                                 

用Reflector查看OjbectQuery对象的GetEnumerator方法,不难找出异常的原因:由于Entity在读取数据的时候使用的是 DbDataReader进行读取,当作为IEnumuerable<T>对象MoveNext进行操作时,只是使用 DbDataReader进行一次Read操作,因此当查询的结果未完全读取完的时候,数据库连接一直被占用,当再次进行查询操作时便回出现上面的异常。

至于解决方案,网上很容易就能找到:

  • 一种方案就是在数据库连接串中设置MultipleActiveResultSets=True,但需要注意的是该方案只适合Sql Server 2005以及以上版本,所以什么Sql Server 2000之类设置也没用。
  • 还一种方案就比较直接,别用什么AsEnumerable之类(直接将查询结果丢进foreach),尽量使用ToArray、ToList方法,什么说 Entity用ToList方法会有效率问题啥的,别乱说先,Reflector很好用的,看看源代码先,别随便整个测试啥的就说Entity这样慢那样 不行。
  • 转自http://www.kwstu.com/ArticleView/guandebao_2013917144342904
 

 

 
 

 

 

使用codefirst查询当然是必不可少的,但有时不小心可能很简单的查询也会导致异常。

下面用codefirst做个示例简单演示下异常发生的场景:

1
2
3
4
5
6
7
var attendlist = db.Database.SqlQuery<AttendCountViewModel>("select STUDENTID,SUM(isnull(REALHOUR,0)) As REALHOUR from ATTEND_LIST where ATTENDDATE=‘" + geDate + "‘ and GOTIME is not null and GETOFFTIME is not null group by STUDENTID");
 
            foreach (AttendCountViewModel item in attendlist)
            {
                STUDENT_INFO student = db.STUDENT_INFO.Find(item.STUDENTID);
                db.SaveChanges();
            }

逻辑比较简单,就是首先查询一个AttendCountViewModel列表,并当作IEnumerable<AttendCountViewModel>对象进行读取的时候,在里面再进行查询操作,不出意外的话,程序会毫不留情的报错:

已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭。                                                                 

用Reflector查看OjbectQuery对象的GetEnumerator方法,不难找出异常的原因:由于Entity在读取数据的时候使用的是 DbDataReader进行读取,当作为IEnumuerable<T>对象MoveNext进行操作时,只是使用 DbDataReader进行一次Read操作,因此当查询的结果未完全读取完的时候,数据库连接一直被占用,当再次进行查询操作时便回出现上面的异常。

至于解决方案,网上很容易就能找到:

  • 一种方案就是在数据库连接串中设置MultipleActiveResultSets=True,但需要注意的是该方案只适合Sql Server 2005以及以上版本,所以什么Sql Server 2000之类设置也没用。
  • 还一种方案就比较直接,别用什么AsEnumerable之类(直接将查询结果丢进foreach),尽量使用ToArray、ToList方法,什么说 Entity用ToList方法会有效率问题啥的,别乱说先,Reflector很好用的,看看源代码先,别随便整个测试啥的就说Entity这样慢那样 不行。

以上是关于code first提示已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭解决方法的主要内容,如果未能解决你的问题,请参考以下文章

[EF] - "已有打开的与此 Command 相关联的 DataReader,必须首先将它关闭" 之解决

EntityFramework Core Code First 已有数据库

ef6 code first,对已有数据库如何执行迁移

EF应用一:Code First模式

codeblocks怎样导入已有项目?为啥每次打开都是main.c文件?

ef core code first 模式提示"可能会导致循环或多重级联路径"问题