2

我创建了一个 MVC 4 并使用 NHibernate 来持久化模型并使用流利的 nhibernate 对其进行映射。该实体具有“名称”属性并以这种方式映射:

 Map(x => x.Name).Not.Nullable().Length(100); 

我创建了一个用于查看对象列表的表格,并让我可以编辑、查看对象详细信息并删除它们。当我删除对象时,视图层将模型 ID 回发到相应的控制器,并且控制器通过存储库对象尝试删除该对象。

 [HttpPost]
    public ActionResult DeleteElement(Element element)
    {
        Element deletedElement = repository.Delete(element);
        TempData["message"] = string.Format("{0} has been deleted.",deletedElement.Name);
        return RedirectToAction("Index");
    }

部分表格视图:

<td>
            @using (Html.BeginForm("DeleteMenu", "Admin"))
            { 
                @Html.Hidden("ID", item.ID)

                <input type="submit" value="Delete"/>
            }
        </td>

因此,视图仅将 elemntID 回发给控制器。元素对象只有它的 ID。并且它的所有属性都是空的。当由于名称属性为空而试图删除对象时,存储库中的会话对象无法删除该对象,因为名称字段为空。

错误消息:

非空属性引用空值或瞬态值 Element.Name

如果我只删除一个对象并拥有主键,为什么 nHibernate 关心其他字段是否为空?以及如何仅删除带有 ID 的对象?

 public IQueryable<T> GetAll()
    {
        return session.Query<T>();
    }

    public IQueryable<T> Get(Expression<Func<T, bool>> predicate)
    {
        return GetAll().Where(predicate);
    }


 public void Delete(T entity)
        {
            session.Delete(entity);
        }
4

1 回答 1

2

首先从 NHibernate 中检索实例,并将用作传递给Delete.

发生的事情是您在 NHibernate 的权限范围之外创建一个对象(在 MVC 模型绑定中)。由于您在 HTML 表单中只指定了 ID,因此当模型绑定器完成时,模型的属性也全部为空。

当您将此对象传递给 NHibernate 时,它​​会注意到会话没有观察到它并尝试附加它,这将看到许多脏属性值(所有空值),因此尝试首先刷新对它的更改。

您的操作应如下所示:

[HttpPost]
public ActionResult DeleteElement(int id)
{   var element = repository.Get(e => e.Id == id).First();
    repository.Delete(element);

    TempData["message"] = string.Format("{0} has been deleted.",deletedElement.Name);
    return RedirectToAction("Index");
}

我建议向您的存储库添加一个Load由 NHibernate 包装的方法。Load很有用,因为它使用您指定的 ID 创建对象的观察实例,但在您访问 ID 以外的属性之前实际上不会访问数据库。这对于您知道对象存在但只需要一个指针(例如删除实体或将其添加到关系中)的情况非常有用。

更新的存储库:

public IQueryable<T> GetAll()
{
    return session.Query<T>();
}

public IQueryable<T> Get(Expression<Func<T, bool>> predicate)
{
    return GetAll().Where(predicate);
}

public T DeferredGet(int id) // I like to call it DeferredGet, you can call it Load or whatever you want
{
       return session.Load<T>(id);
}

public void Delete(T entity)
{
    session.Delete(entity);
}

然后是您更新的操作:

[HttpPost]
public ActionResult DeleteElement(int id)
{   var element = repository.DeferredGet(id); // will not actually hit the database, saving you a query.
    repository.Delete(element); // deletes the element normally.

    TempData["message"] = string.Format("{0} has been deleted.",deletedElement.Name);
    return RedirectToAction("Index");
}
于 2013-10-21T18:59:47.433 回答