12

想象一下,我有以下内容:

private IEnumerable MyFunc(parameter a)
{
   using(MyDataContext dc = new MyDataContext)
   {
      return dc.tablename.Select(row => row.parameter == a);
   }
}

private void UsingFunc()
{
   var result = MyFunc(new a());

   foreach(var row in result)
   {
      //Do something
   }
}

根据文档,linq 执行将推迟到我实际枚举结果,该结果出现在 foreach 的行中。然而 using 语句应该强制在调用 MyFunct() 结束时可靠地收集对象。

实际发生了什么,处置器何时运行和/或结果运行?

我唯一能想到的是延迟执行是在编译时计算的,所以编译器将实际调用移动到 foreach 的第一行,导致 using 正确执行,但直到 foreach 行才运行?有大神可以帮忙吗?

编辑:注意:这段代码确实有效,我只是不明白如何。

我做了一些阅读,并在我的代码中意识到我调用了 ToList() 扩展方法,该方法当然会枚举结果。勾选答案的行为对于回答的实际问题是完全正确的。

很抱歉有任何混淆。

4

2 回答 2

13

我希望这根本行不通。是Select延迟的,因此此时没有消耗任何数据。但是,由于您已经处置了数据上下文(在离开之前MyFunc),因此它将永远无法获取数据。更好的选择是将数据上下文传递方法中,以便消费者可以选择生命周期。另外,我建议返回IQueryable<T>,以便消费者可以“组合”结果(即添加///OrderBy等,并使其影响最终查询):SkipTakeWhere

// this could also be an instance method on the data-context
internal static IQueryable<SomeType> MyFunc(
    this MyDataContext dc, parameter a)
{
   return dc.tablename.Where(row => row.parameter == a);
}

private void UsingFunc()
{
    using(MyDataContext dc = new MyDataContext()) {
       var result = dc.MyFunc(new a());

       foreach(var row in result)
       {
           //Do something
       }
    }
}

更新:如果您(评论)不想推迟执行(即您不希望调用者处理数据上下文),那么您需要评估结果。您可以通过调用.ToList().ToArray()在结果上缓冲值来做到这一点。

private IEnumerable<SomeType> MyFunc(parameter a)
{
   using(MyDataContext dc = new MyDataContext)
   {
      // or ToList() etc
      return dc.tablename.Where(row => row.parameter == a).ToArray();
   }
}

如果你想在这种情况下保持延迟,那么你需要使用“迭代器块”:

private IEnumerable<SomeType> MyFunc(parameter a)
{
   using(MyDataContext dc = new MyDataContext)
   {
      foreach(SomeType row in dc
          .tablename.Where(row => row.parameter == a))
      {
        yield return row;
      }
   }
}

现在在不传递数据上下文的情况下将其推迟。

于 2009-01-19T06:22:04.043 回答
8

我刚刚在此处发布了针对此问题的另一个延迟执行解决方案,包括以下示例代码:

IQueryable<MyType> MyFunc(string myValue)
{
    return from dc in new MyDataContext().Use()
           from row in dc.MyTable
           where row.MyField == myValue
           select row;
}

void UsingFunc()
{
    var result = MyFunc("MyValue").OrderBy(row => row.SortOrder);
    foreach(var row in result)
    {
        //Do something
    }
}

扩展方法本质上Use()就像一个延迟using块。

于 2009-07-23T08:38:20.557 回答