asp.net-mvc-3 - 发布表单时 Controller 中的模型绑定(bind) - 导航属性不会自动加载

标签 asp.net-mvc-3 entity-framework-4.1 model-binding

我使用的是 Entity Framework 4.2 版。我的小测试应用程序中有两个类:

public class TestParent
{
    public int TestParentID { get; set; }
    public string Name { get; set; }
    public string Comment { get; set; }

    public virtual ICollection<TestChild> TestChildren { get; set; }
}

public class TestChild
{
    public int TestChildID { get; set; }
    public int TestParentID { get; set; }
    public string Name { get; set; }
    public string Comment { get; set; }

    public virtual TestParent TestParent { get; set; }
}

用数据库中的数据填充对象效果很好。所以我可以在我的代码中使用 testParent.TestChildren.OrderBy(tc => tc.Name).First().Name 等。

然后我为我的 testParents 构建了一个标准的编辑表单。 Controller 看起来像这样:

public class TestController : Controller
{
    private EFDbTestParentRepository testParentRepository = new EFDbTestParentRepository();
    private EFDbTestChildRepository testChildRepository = new EFDbTestChildRepository();

    public ActionResult ListParents()
    {
        return View(testParentRepository.TestParents);
    }

    public ViewResult EditParent(int testParentID)
    {
        return View(testParentRepository.TestParents.First(tp => tp.TestParentID == testParentID));
    }

    [HttpPost]
    public ActionResult EditParent(TestParent testParent)
    {
        if (ModelState.IsValid)
        {
            testParentRepository.SaveTestParent(testParent);
            TempData["message"] = string.Format("Changes to test parents have been saved: {0} (ID = {1})",
                                                        testParent.Name,
                                                        testParent.TestParentID);
            return RedirectToAction("ListParents");
        }
        // something wrong with the data values
        return View(testParent);
    }
}

当表单回发到服务器时,模型绑定(bind)似乎运行良好 - 即 testParent 看起来没问题(id、名称和注释设置如预期)。但是,导航属性 TestChildren 仍为 NULL。

我想这并不奇怪,因为模型绑定(bind)只是提取从浏览器发送的表单值,并将它们推送到 TestParent 类的对象中。然而,填充 testParent.TestChildren 需要立即往返于 Entity Framework 负责的数据库。并且 EF 可能不会参与绑定(bind)过程。

然而,当我调用 testParent.TestChildren.First() 时,我期待延迟加载开始。相反,这会导致 ArgumentNullException。

是否需要在模型绑定(bind)后以特殊方式标记对象,以便 Entity Framework 进行延迟加载?我怎样才能做到这一点?

显然,我可以使用第二个存储库 testChildRepository 手动检索子项。但是 (a) 感觉不对,并且 (b) 导致我的存储库设置方式出现问题(每个存储库都使用自己的 DBContext - 这是一个我还没有设法解决的问题)。

最佳答案

为了让您的子集合延迟加载,必须满足两个要求:

  • 父实体必须附加到 EF 上下文
  • 您的父实体必须是延迟加载代理

如果您通过上下文从数据库加载父实体(并且您的导航属性是虚拟以允许创建代理),则这两个要求都得到满足。

如果您不从数据库加载实体而是手动创建它,您可以通过使用适当的 EF 函数来实现相同的目的:

var parent = context.TestParents.Create();
parent.TestParentID = 1;
context.TestParents.Attach(parent);

使用 Create 而不是 new 在这里很重要,因为它会创建所需的延迟加载代理。然后您可以访问子集合,ID = 1 的父集合的子集合将被延迟加载:

var children = parent.TestChildren; // no NullReferenceException

现在,默认的 modelbinder 对这些特定的 EF 函数一无所知,只会用 new 实例化父级,也不会将其附加到任何上下文。这两个要求都没有满足,延迟加载无法工作。

您可以编写自己的模型绑定(bind)器以使用 Create() 创建实例,但这可能是最糟糕的解决方案,因为它会使您的 View 层非常依赖 EF。

如果您在模型绑定(bind)后需要子集合,在这种情况下我会通过显式加载来加载它:

// parent comes as parameter from POST action method
context.TestParents.Attach(parent);
context.Entry(parent).Collection(p => p.TestChildren).Load();

如果您的上下文和 EF 隐藏在存储库后面,您将需要一个新的存储库方法,例如:

void LoadNavigationCollection<TElement>(T entity,
    Expression<Func<T, ICollection<TElement>>> navigationProperty) 
    where TElement : class
{
    _context.Set<T>().Attach(entity);
    _context.Entry(entity).Collection(navigationProperty).Load();
}

...其中 _context 是存储库类的成员。

但是,正如 Darin 提到的,更好的方法是绑定(bind) ViewModel,然后根据需要将它们映射到您的实体。然后您可以选择使用 Create() 实例化实体。

关于asp.net-mvc-3 - 发布表单时 Controller 中的模型绑定(bind) - 导航属性不会自动加载,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/8883264/

相关文章:

asp.net-mvc - asp.net mvc Beta1中的模型绑定(bind)

asp.net-mvc - MVC基于角色的路由

c# - .NET MVC 应用程序中所有 HTTP GET 操作的 OnActionExecuted 代码

asp.net-mvc - MVC 4 忽略 DefaultModelBinder.ResourceClassKey

android - 代码 fragment 的 Mvvmcross 绑定(bind)子项

c# - Entity Framework 以不确定的方式未经许可删除有效内容?

iis - asp.net mvc 3的必要要求?

使用 Enter 键时,HTML 表单不包含表单提交按钮名称

entity-framework-4.1 - Entity Framework - 使用 Code First 的复杂模型

entity-framework - EF 中的问题,映射片段,无默认值且不可为空