我注释掉了从我的 ObjectContext 类的所有构造函数重载启用延迟加载的代码行,但是当我迭代导航属性时,迭代成功。为什么?
这是相关的代码。
public MyExpensesEntities() :
base("name=MyExpensesEntities", "MyExpensesEntities")
{
// this.ContextOptions.LazyLoadingEnabled = true;
OnContextCreated();
}
static void Main(string[] args)
{
AddExpenses();
Console.WriteLine("Lazy loading is {0}.",
_context.ContextOptions.LazyLoadingEnabled ?
"enabled": "disabled");
PrintCategorywiseExpenses();
_context.Dispose();
Console.WriteLine("Press any key to exit...");
Console.ReadKey();
}
static void PrintCategorywiseExpenses()
{
foreach (var cateogry in _context.Categories)
{
Console.WriteLine
("Category: {0}\n----------------",
cateogry.CategoryName);
foreach (var e in cateogry.Expenses)
Console.WriteLine
("\tExpense: {0}\tAmount: {1}",
e.Particulars, e.Amount.ToString("C"));
}
Console.WriteLine();
}
最佳答案
您应该知道,通过对导航属性执行 foreach
,您明确要求 EF 为您加载它,仅仅因为您关闭延迟加载并不意味着您不能显式加载他们以后。因此,一旦它被禁用,您仍然可以根据需要显式加载相关数据,这正是您使用 foreach
时发生的情况。 “延迟加载” 的另一个术语实际上是“隐式延迟加载”,它不会阻止我们做“显式延迟加载”
基本上,这行代码:
foreach (var cateogry in _context.Categories)
几乎等同于:
_context.Categories.AsEnumerable()
要了解禁用延迟加载如何影响您的代码,请参见下面的示例。
在此示例中,我禁用它并读取您的一个类别,然后处理 ObjectContext 并在 Expenses 上执行 foreach:
Category category;
using (MyEntities context = new MyEntities()) {
category = _context.Categories.First();
}
foreach (var e in cateogry.Expenses)
Console.WriteLine("Expense:{0}", e.Amount.ToString("C"));
}
延迟加载已禁用:
没有任何反应,代码永远不会进入 foreach,因为 cateogry.Expenses.Count == 0
延迟加载已启用:
EF 尝试延迟加载 Expenses,并且由于 ObjectContext 已被释放(通过超出 using 范围),您将获得 System.ObjectDisposedException 包含此消息:
ObjectContext 实例已被释放,不能再用于需要连接的操作。
关于c# - 即使在 ObjectContext.ContextOptions.LazyLoadingEnabled = false 时也会延迟加载导航属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3990962/