Свободное отображение NHibernate IDictionaryумным способом - PullRequest
5 голосов
/ 30 декабря 2010

Учитывая эти классы:

using System.Collections.Generic;

namespace FluentMappingsQuestion
{
    public class Entity
    {
        public virtual int Id { get; set; }
        public virtual IDictionary<string, Property> Properties { get; set; }
    }

    public class Property
    {
        public virtual Entity OwningEntity { get; set; }
        public virtual string Name { get; set; }
        public virtual int Value { get; set; }
        public virtual decimal OtherValue { get; set; }
    }
}

Как я могу отобразить их с помощью NHibernate (желательно с беглым вкусом), чтобы сделать это возможным:

[Test]
public void EntityPropertyMappingTest()
{
    using (var session = _factory.OpenSession())
    {
        var entity = new Entity();

        // (#1) I would *really* like to use this
        entity.Properties["foo"] = new Property { Value = 42, OtherValue = 42.0M };

        session.Save(entity);
        session.Flush();

        // (#2) I would like the entity below to "update itself"
        // on .Save or .Flush. I got it to work with .Load()
        Assert.AreEqual(42, entity.Properties["foo"].Value);
        Assert.AreEqual(42.0M, entity.Properties["foo"].OtherValue);
        Assert.AreEqual("foo", entity.Properties["foo"].Name);
        Assert.AreEqual(entity, entity.Properties["foo"].Owner);
    }
}

Мне почти удалось сделать это с этими сопоставлениями:

// class EntityMap : ClassMap<Entity>
public EntityMap()
{
    Id(x => x.Id);
    HasMany(x => x.Properties)
    .Cascade.AllDeleteOrphan()
    .KeyColumn("EntityId")
    .AsMap(x => x.Name);
}

// class PropertyMap : ClassMap<Property>
public PropertyMap()
{
    Id(x => x.Id);
    References(x => x.OwningEntity).Column("EntityId");
    Map(x => x.Name).Length(32);
    Map(x => x.Value);
{

Проблемы у меня:

  • Если я сделаю Entity.Properties .Inverse(), он начнет ломаться
  • Если я не сделаю это .Inverse(), тогда NHibernate сделает: INSERT(Entity), INSERT(Property), UPDATE(Property) вместо просто INSERT(Entity), INSERT(Property)
  • Если я сделаю Property.Name .Not.Nullable(), он начнет ломаться
  • Если я не сделаю это .Not.Nullable(), у меня есть дыра в моей схеме БД

Как мне изменить мои отображения?

1 Ответ

4 голосов
/ 13 января 2011

Я работал над этим, указав это отображение:

HasMany<Property>(Reveal.Member<Entity>("InternalProperties"))
  .AsMap(p => p.Name)
  .Cascade.AllDeleteOrphan()
  .Inverse();

и создав два свойства типа IDictionary<string, Property>: Properties и InternalProperties.Первый из них является прокси-словарем над вторым и имеет дело с установкой свойств OwningEntity и Name для записей Property.

...