如何告诉NHibernate只保存已更改的属性

本文关键字:属性 保存 何告诉 NHibernate | 更新日期: 2023-09-27 17:58:47

我用NHibernate将类Person映射到数据库。我从数据库加载对象并将其发送到不同的客户端。第一个客户端将修改Name和Country属性。第二个客户端将只修改Name属性。然后两者都将修改后的对象返回到服务器。当我从第一个客户端保存数据时,然后正确保存,名称和国家都更新了。当我从第二个客户端保存数据时,我遇到了问题。它覆盖了来自第一个客户端的数据,并保存了国家的新名称和初始值。

如何告诉NHibernate只保存Name值而不覆盖Country值?

public class Person
{
    public string Name { get; set; }
    public string Country { get; set; }
}
public static List<Person> GetEntities()
{
    var factory = CreateSessionFactory();
    using (ISession session = factory.OpenSession())
    {
        return session.CreateCriteria<Person>().List<Person>();                
    }
}
public static void SaveEntities(List<Person> entities)
{
    var factory = CreateSessionFactory();
    using (ISession session = factory.OpenSession())
    {
         using (var t = session.BeginTransaction())
         {
             foreach (var person in entities)
             {
                 session.Merge(person);
             }
             t.Commit();
        }
    }
}

p.S:很抱歉我的英语不好

如何告诉NHibernate只保存已更改的属性

实际上,您可以告诉NHibernate使用Dynamic Update专门更新"脏"字段。

更多信息:http://ayende.com/blog/3946/nhibernate-mapping-concurrency

这是一个并发问题。第二个客户端在读取数据后并不知道数据发生了更改,因此他们的更改会覆盖第一个客户端的更改。这可以在NHibernate中通过几种方法之一进行处理,其中最常见的方法是使用版本列。

这个问题很容易预防,更大的问题是在发生时向用户提供良好的反馈。

答案是:你不能。NH不知道只有名字变了。

您可以通过不允许并发编辑来避免这种情况。例如,NH的乐观锁定机制。第二个客户端将得到一个CCD_ 2。

如果两个客户端的编辑实际上不在同一时间(但都基于相同的对象状态),则需要确保第二个客户端在编辑前获得第一个客户端的更改。例如,通过在打开编辑器之前检索实际状态,或者通过从服务器发送更改的通知。

如果你想保持并发编辑,你还有很多工作要做。客户端需要提供实际更改的信息。然后只需要复制这些值。这是一项艰苦的工作。然后,您可能仍然会遇到这样合并后的值不匹配的问题。

正如Jamie Ide正确指出的,您所经历的是并发问题,而不是映射问题。。

当您创建对象的nhibernate映射时,一旦刷新会话,这些对象中存在的任何数据都将保存(或更新)在数据库中。您不能指定要更新的单个字段,要么全部更新,要么什么都不更新。

还有一点(不相关的),您的示例代码正在实体的每个操作上实例化一个新的会话工厂。这通常是个坏主意,因为会话工厂的创建成本很高。您最好通过全局上下文(singleton)管理会话工厂,创建一次,并在需要时生成会话(它们是较轻的对象)。