NHibernate,将DB中的空引用值保存为0(零)

时间:2012-03-13 15:34:59

标签: c# nhibernate legacy-database tuplizer

我发现这个tuplizer的例子在保存空关系时保存0。这是必需的,因为我正在处理遗留数据库模式上的应用程序。

我在这里尝试了tuplizer:http://nhforge.org/blogs/nhibernate/archive/2011/01/28/how-to-use-0-instead-of-null-for-foreign-keys.aspx

在该示例中,我得到了ProxyFactory的nullreferenceexception。然后我在这里找到了代码的更新:https://bitbucket.org/jfromaniello/hotgazpachoeg/changeset/87ac41c473ae

但是,这对我也不起作用。在最后一个方法中,SetPropertyValues(描述​​为脏黑客3,在从DB读取对象时使用),我在这部分得到一个nullref异常,if(typeof(IEntity),当读取一个不相关的对象(不是一个Sample)时/ p>

我的映射如下(简化):

   Table("ej_sample");
        Not.LazyLoad();
        Id(s => s.Id, "sampleID").GeneratedBy.Native();
        References<Sample>(s => s.ParentSample, "parentSampleID").NotFound.Ignore();

当没有此类对象时,parentSampleID列必须为0.

我想,我只需要对插入和更新进行肮脏的黑客攻击(可能只在我的情况下插入)。

在插入时,我想创建一个伪代理,但[2]中的代码从db加载实体(可能使用Null对象?!)。

插入脏黑客:

        public override object[] GetPropertyValuesToInsert(object entity, IDictionary mergeMap, ISessionImplementor session) {
        var values = base.GetPropertyValuesToInsert(entity, mergeMap, session);

        //dirty hack 1
        for(int i = 0; i < values.Length; i++) {
            if(values[i] == null && typeof(IEntity).IsAssignableFrom(getters[i].ReturnType)) {
                values[i] = ((ISession)session).Load(getters[i].ReturnType, 0);
            }
        }
        return values;
    }

我尝试创建假代理而不是执行上述操作:

        public override object[] GetPropertyValuesToInsert(object entity, IDictionary mergeMap, ISessionImplementor session) {
        var values = base.GetPropertyValuesToInsert(entity, mergeMap, session);

        //dirty hack 1
        for(int i = 0; i < values.Length; i++) {
            if(values[i] == null && typeof(IEntity).IsAssignableFrom(getters[i].ReturnType)) {
                //values[i] = ((ISession)session).Load(getters[i].ReturnType, 0);
                values[i] = CreateFakeProxy(i);
            }
        }
        return values;
    }

    private object CreateFakeProxy(int i) {
        object proxy;
        using(var sessionImplementor = _sessionFactory.OpenSession()) {
            proxy = _sessionFactory
                .GetEntityPersister(getters[i].ReturnType.FullName)
                .CreateProxy(0, (ISessionImplementor)sessionImplementor);
        }
        return proxy;
    }

然后我在_sessionfactory上得到一个nullref异常,它在ctor中设置:

        private readonly ISessionFactoryImplementor _sessionFactory;

    public NullableTuplizer(EntityMetamodel entityMetamodel, PersistentClass mappedEntity)
        : base(entityMetamodel, mappedEntity) {
            _sessionFactory = entityMetamodel.SessionFactory;

    }

任何想法如何实现这一目标?

1 个答案:

答案 0 :(得分:1)

简单的解决方案是添加以下preinsert和preupdate侦听器。

public class NullToZeroEventListener : AuditEventListener, IPreInsertEventListener, IPreUpdateEventListener
{
    public bool OnPreInsert(PreInsertEvent @event) {
        ZeroNullIds(@event.State, @event.Persister.PropertyNames);
        return false;
    }

    public bool OnPreUpdate(PreUpdateEvent @event) {
        ZeroNullIds(@event.State, @event.Persister.PropertyNames);
        return false;
    }

    protected internal void ZeroNullIds(Object[] state, string[] propertyNames) {
        for(int i = 0; i < propertyNames.Length; i++) {
            if(state[i] != null || propertyNames[i].EndsWith("ID")) continue;
            state[i] = 0;
        }
    }
}

在映射中一定要忽略0 ids,例如: References<User>(s => s.User, "userID").NotFound.Ignore().LazyLoad();

在你的sessionfactory中,为preinsert和preupdate事件添加一个监听器(首先显示在这里):

           .ExposeConfiguration(c =>
            {
                if(!c.EventListeners.PreInsertEventListeners.Any()) {
                    c.AppendListeners(ListenerType.PreInsert, new IPreInsertEventListener[] { new NullToZeroEventListener() });
                }
            });