5

我正在使用 NHibernate 拦截器将有关更新/插入/删除的信息记录到我的各种实体中。

记录的信息中包括实体类型和已修改实体的唯一 ID。<generator class="identity">唯一 ID在 NHibernate 映射文件中标记为 a 。

明显的问题是当使用 IInterceptor.OnSave() 记录插入操作时,尚未分配实体的 ID。

如何在记录审计信息之前获取插入实体的 ID?

(我已经查看了 NHibernate Listeners PostSave 事件,但无法让它们与正在使用的 Spring.net 配置一起工作,所以如果可能的话,我想坚持使用拦截器)

代码:

    // object id parameter is null...
    public override bool OnSave(object entity, object id, object[] state, 
        string[] propertyNames, IType[] types)
    {            
        AddAuditItem(entity, INSERT);
        return false;            
    }
4

2 回答 2

10

我通过向我的拦截器类添加一个列表来解决这个问题,该类在OnSave实现期间填充了对象。

PostFlush实现中,列表被迭代,每个元素都被审计为插入。此列表中的对象已被持久化PostFlush(),因此已生成 ID。

这似乎工作正常,但如果指出任何潜在的陷阱,我将不胜感激:-)

public class AuditInterceptor : EmptyInterceptor
{       
    // To hold inserted items to be audited after insert has been flushed
    private IList<object> insertItems = new List<object>();

    public override void PostFlush(System.Collections.ICollection entities)
    {            
        foreach (var entity in insertItems)
        {
            AddAuditItem(entity, INSERT);
        }
        insertItems.Clear();

        base.PostFlush(entities);
    }

    public override bool OnSave(object entity, object id, object[] state, 
        string[] propertyNames, IType[] types)
    {            
        var auditable = entity as IAuditable;
        if (auditable != null) 
            insertItems.Add(entity);

        return false;            
    }
}
于 2009-05-22T15:35:21.740 回答
0

试试这个OnFlushDirty方法..或者也许PostFlush

编辑:另外,你可以发布你的代码吗?你不把 Id 作为参数OnSave吗?

于 2009-05-21T10:41:11.313 回答