2012-03-13 69 views
1

我发现了一个的tuplizer的这个例子中节省空当的关系做保存0。由于我正在处理遗留数据库模式上的应用程序,因此这是必需的。NHibernate的,保存为0(零)DB空的参考价值

我想这里的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,阅读从数据库对象时使用),我得到这部分nullref例外,如果(typeof运算(IEntity),读一本不相关的对象(而不是样本)

我的映射如下(简化):

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

当该对象不存在的parentSampleID列必须是0

我想,我只需要做上插入和更新脏黑客。 (可能在我的情况下只能插入)

On inser t,我想创建一个伪代理,但[2]中的代码从db加载实体(可能使用空对象?!)。

插入肮脏的黑客:

 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例外:

 private readonly ISessionFactoryImplementor _sessionFactory; 

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

    } 

任何想法如何做到这一点?

+0

您将与ISessionImplementor无关的'ISession'强制转换为'ISessionImplementor'。更安全的是'session.GetSessionImplementor()' – Firo 2012-03-13 16:04:48

回答

1

简单的解决方法是添加以下preinsert和更新前的监听器。

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的ID,例如: References<User>(s => s.User, "userID").NotFound.Ignore().LazyLoad();

在你的SessionFactory,添加监听两个preinsert和更新前的事件(这里先出):

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

提示:“Pre”事件监听器正式不是用来修改实体的。改用DefaultSaveEventListener。我知道互联网上充斥着修改实体的“Pre”侦听器示例,但Hibernate开发人员回答说这种用法不受支持,因此可能会产生新问题。 – JustAMartin 2012-09-28 14:44:19