实际上它不一定是IDataReader
。
我有一个像这样的函数:
public IEnumerable<MyClass> GetObjects()
{
IDataReader dr = GetSomeDataReader();
while (dr.Read())
{
yield return new MyClass(dr);
}
CloseDBConnections();
}
这样做很好,直到我像这样重构它:
public IEnumerable<MyClass> GetObjects()
{
IDataReader dr = GetSomeDataReader();
try
{
return ProcessReader(dr);
} finally {
CloseDBConnections();
}
}
public IEnumerable<MyClass> ProcessReader(IDataReader dr)
{
while (dr.Read())
{
yield return new MyClass(dr);
}
}
这不起作用,因为执行CloseDBConnections()
时枚举尚未处理。
从.ToList()
返回时调用GetObjects
实际执行枚举,但到那时连接已被破坏,IDataReader
失败。
在我的实例中CloseDBConnections
无法在新的ProcessReader
函数中调用,因为IDataReader
可能来自其他来源(在此实例中重新分解的整个点)
是否有合理的解决方法或者我是否必须复制枚举代码?
我尝试将ProcessReader
作为yield return
而不是return
,但这不起作用,因为C#(可以理解)认为我正在尝试添加IEnumerable
到IEnumerable
!
答案 0 :(得分:3)
不漂亮,但这应该有效。
public IEnumerable<MyClass> GetObjects()
{
IDataReader dr = GetSomeDataReader();
try
{
foreach (var result in ProcessReader(dr))
{
yield return result;
}
} finally {
CloseDBConnections();
}
}
答案 1 :(得分:3)
如何通过回调在CloseDBConnections
中呼叫ProcessReader
?
public IEnumerable<MyClass> GetObjects()
{
return ProcessReader(GetSomeDataReader(), CloseDBConnections);
}
public IEnumerable<MyClass> ProcessReader(IDataReader dr, Action OnFinished)
{
try
{
while (dr.Read())
yield return new MyClass(dr);
}
finally
{
if (OnFinished != null)
OnFinished();
}
}