3

我正在尝试使用流利的 nhibernate 配置操作锁定。

那里有很多信息,但似乎没有一个适合我所处的场景。我的班级和地图如下(为简洁起见进行了编辑):

实体:

public class EmailGroup : CRUDDomainObject<EmailGroup>
{        
    public virtual string Id { get; set; }
    public virtual MailServer Server { get; set;}
    public virtual string FromAddress { get; set;}
    public virtual string ToAddress { get; set;}
    public virtual long Version { get; set; }    
}

地图:

public class EmailGroupMap : ClassMap<EmailGroup>
{
    public const string TABLE_ID = "EMAILGROUP";

    public const string FIELD_ID = "EMAILID";
    public const string FIELD_MAIL_SERVER = "MAILSERVID";
    public const string FIELD_FROM_ADDRESS = "EMLFROM";
    public const string FIELD_TO_ADDRESS = "EMLTO";
    public const string FIELD_VERSION = "VERSION";

    public EmailGroupMap()
    {
        Table(TABLE_ID);

        Id(x => x.Id)
            .Column(FIELD_ID)
            .Not.Nullable()
            .GeneratedBy.Assigned()
            .Length(12);

        References(x => x.Server)
            .Column(FIELD_MAIL_SERVER)
            .NotFound.Ignore();

        Map(x => x.FromAddress)
            .Column(FIELD_FROM_ADDRESS)
            .Not.Nullable()
            .Length(120);

        Map(x => x.ToAddress)
            .Column(FIELD_TO_ADDRESS)
            .Not.Nullable()
            .Length(1000);

        Version(X => X.Version)
            .Column(FIELD_VERSION)
            .Generated.Always()
            .UnsavedValue("0")
            .Access.Property();

        DynamicUpdate();

        OptimisticLock.Version();
    }
}

在我看来一切都很好,但是当我加载实体并对其进行修改时,版本号不会增加。同样,如果我在会话打开时手动增加版本,我不会收到 StaleObjectException。

这个配置对更有经验的人来说是否有效?如果是这样,我还能错过什么?

更新:

在实施数据库管理的时间戳之后,版本列(当然)正在递增。然而 NHibernate 不会将该行视为乐观锁定。我从 SQL 服务器捕获更新查询以检查 where 子句(为简洁起见截断):

exec sp_executesql N'UPDATE [EMAILGROUP] 
SET [EMLDESC] = @EMLDESC, [MAILSERVID] = @MAILSERVID, [EMLFROM] = @EMLFROM, [EMLTO] = @EMLTO, [EMLCC] = @EMLCC, [EMLBCC] = @EMLBCC 
WHERE [EMAILID] = @EMAILID'
4

2 回答 2

2

和 SQL Server最典型的场景Version(不确定这是不是你的情况)是 sql 类型timestamp(过时的)或更好rowversion的。这应该映射到 C# byte[]。因此,这些更改应该可以解决它:

1)服务器上的版本列必须是类型rowversion(或timestamp)。这样的列会自动更新与当前行相关的任何更改。每个表只能存在一个这样的列

2) 实体应如下所示

public class EmailGroup : CRUDDomainObject<EmailGroup>
{        
   ...
   public virtual byte[] Version { get; set; }  

3) 流畅的映射代码应保持原样。做背后的技巧应该是一个流利的映射器的工作。我们需要实现的是这样的:

<version name="Version" generated="always" unsaved-value="null"  type="BinaryBlob">
    <column name="Version" not-null="false" sql-type="timestamp"/>
</version>

请在这里查看更多信息:http: //ayende.com/blog/3946/nhibernate-mapping-concurrency

于 2013-07-03T04:11:59.707 回答
2

你为什么指定Generated.Always()?这告诉 NHibernate 这不是一个真正的列,而是由数据库计算的。文档: http: //nhibernate.info/doc/nh/en/index.html#mapping-generated

删除它,它应该可以工作。

于 2013-07-03T07:29:21.253 回答