流畅的 NHibernate 自动映射每个抽象层次结构表/每个具体子类表

发布于 2024-12-22 18:14:58 字数 3858 浏览 2 评论 0原文

我有课程

public abstract class Content : IContent
{
    public virtual Guid Id { get; protected set; }
    public virtual IPage Parent { get; set; }
    public virtual DateTime Created { get; set; }
    /* ... */
}
public abstract class Page : Content, IPage
{
    public virtual string Slug { get; set; }
    public virtual string Path { get; set; }
    public virtual string Title { get; set; }
    /* ... */
}

public class Foo : Page, ITaggable
{
    // this is unique property 
    // map to joined table
    public virtual string Bar { get; set; }

    // this is a unique collection
    public virtual ISet<Page> Related { get; set; }

    // this is "shared" property (from ITaggable)
    // map to shared table
    public virtual ISet<Tag> Tags { get; set; }
}

,因此我想要下表。我尝试过实现大量不同的 IConvention,但即使是层次结构映射(每个抽象层次结构表/每个具体子类表)似乎也失败了。

Content
    Id
    Type (discriminator)
    ParentId
    Created
    Slug
    Path
    Title

Content_Tags (Tags from ITaggable)
    ContentId
    TagId

Content$Foo
    Bar

Content$Foo_Related
    ParentFooId
    ChildPageId

我已经有了丑陋的、工作流畅的映射,但我想摆脱一些丑陋的东西

public class ContentMapping : ClassMap<Content>
{
    public ContentMapping()
    {
        Table("Content");
        Id(x => x.Id).GeneratedBy.GuidComb();
        References<Page>(x => x.Parent, "ParentId");
        Map(x => x.Created);

        DiscriminateSubClassesOnColumn("Type");
    }
}
public class PageMapping : SubclassMap<Page>
{
    public PageMapping()
    {
        Map(x => x.Slug);
        Map(x => x.Path);
        Map(x => x.Title);
    }
}
public class ConcreteContentMapping<T> : SubclassMap<T> where T : Content, new()
{
    public ConcreteContentMapping() : this(true) { }
    protected ConcreteContentMapping(bool mapJoinTable)
    {
        DiscriminatorValue(typeof(T).FullName);
        MapCommonProperties();

        if(mapJoinTable)
        {
            MapJoinTableWithProperties(CreateDefaultJoinTableName(), GetPropertiesNotFrom(GetContentTypesAndInterfaces().ToArray()));
        }
    }

    private void MapCommonProperties()
    {
        if (typeof(ITagContext).IsAssignableFrom(typeof(T)))
        {
                Map(x => ((ITagContext)x).TagDirectory);
        }
        if (typeof(ITaggable).IsAssignableFrom(typeof(T)))
        {
                HasManyToMany(x => ((ITaggable)x).Tags).Table("Content_Tags").ParentKeyColumn("ContentId").ChildKeyColumn("TagId").Cascade.SaveUpdate();
        }
    }
    /* ... */

    // something I would like to get rid of with automappings...
    protected void MapCollectionProperty(JoinPart<T> table, PropertyInfo p)
    {
        var tableName = ((IJoinMappingProvider)table).GetJoinMapping().TableName + "_" + p.Name;
        var elementType = p.PropertyType.GetGenericArguments()[0];

        var method = table.GetType().GetMethods().Where(m => m.Name == "HasManyToMany")
            .Select(m => new { M = m, P = m.GetParameters() })
            .Where(x => x.P[0].ParameterType.GetGenericArguments()[0].GetGenericArguments()[1] == typeof(object))
            .FirstOrDefault().M.MakeGenericMethod(elementType);

        dynamic m2m = method.Invoke(table, new object[] { MakePropertyAccessExpression(p)});
        m2m.Table(tableName).ParentKeyColumn("Parent" + typeof(T).Name + "Id").ChildKeyColumn("Child" + elementType.Name + "Id");
    }
    protected Expression<Func<T, object>> MakePropertyAccessExpression(PropertyInfo property)
    {
        var param = Expression.Parameter(property.DeclaringType, "x");
        var ma = Expression.MakeMemberAccess(param, property);
        return Expression.Lambda<Func<T, object>>(ma, param);
    }
}

如何通过自动映射获得相同的结果?

I have classes

public abstract class Content : IContent
{
    public virtual Guid Id { get; protected set; }
    public virtual IPage Parent { get; set; }
    public virtual DateTime Created { get; set; }
    /* ... */
}
public abstract class Page : Content, IPage
{
    public virtual string Slug { get; set; }
    public virtual string Path { get; set; }
    public virtual string Title { get; set; }
    /* ... */
}

public class Foo : Page, ITaggable
{
    // this is unique property 
    // map to joined table
    public virtual string Bar { get; set; }

    // this is a unique collection
    public virtual ISet<Page> Related { get; set; }

    // this is "shared" property (from ITaggable)
    // map to shared table
    public virtual ISet<Tag> Tags { get; set; }
}

And as a result I'd like to have the following tables. I've tried implementing tons of different IConventions, but even the hierarchy mappings (table-per-abstract-hierarchy / table-per-concrete-subclass) seem to fail.

Content
    Id
    Type (discriminator)
    ParentId
    Created
    Slug
    Path
    Title

Content_Tags (Tags from ITaggable)
    ContentId
    TagId

Content$Foo
    Bar

Content$Foo_Related
    ParentFooId
    ChildPageId

I already have ugly, working fluent mappings, but I would like to get rid of some ugliness

public class ContentMapping : ClassMap<Content>
{
    public ContentMapping()
    {
        Table("Content");
        Id(x => x.Id).GeneratedBy.GuidComb();
        References<Page>(x => x.Parent, "ParentId");
        Map(x => x.Created);

        DiscriminateSubClassesOnColumn("Type");
    }
}
public class PageMapping : SubclassMap<Page>
{
    public PageMapping()
    {
        Map(x => x.Slug);
        Map(x => x.Path);
        Map(x => x.Title);
    }
}
public class ConcreteContentMapping<T> : SubclassMap<T> where T : Content, new()
{
    public ConcreteContentMapping() : this(true) { }
    protected ConcreteContentMapping(bool mapJoinTable)
    {
        DiscriminatorValue(typeof(T).FullName);
        MapCommonProperties();

        if(mapJoinTable)
        {
            MapJoinTableWithProperties(CreateDefaultJoinTableName(), GetPropertiesNotFrom(GetContentTypesAndInterfaces().ToArray()));
        }
    }

    private void MapCommonProperties()
    {
        if (typeof(ITagContext).IsAssignableFrom(typeof(T)))
        {
                Map(x => ((ITagContext)x).TagDirectory);
        }
        if (typeof(ITaggable).IsAssignableFrom(typeof(T)))
        {
                HasManyToMany(x => ((ITaggable)x).Tags).Table("Content_Tags").ParentKeyColumn("ContentId").ChildKeyColumn("TagId").Cascade.SaveUpdate();
        }
    }
    /* ... */

    // something I would like to get rid of with automappings...
    protected void MapCollectionProperty(JoinPart<T> table, PropertyInfo p)
    {
        var tableName = ((IJoinMappingProvider)table).GetJoinMapping().TableName + "_" + p.Name;
        var elementType = p.PropertyType.GetGenericArguments()[0];

        var method = table.GetType().GetMethods().Where(m => m.Name == "HasManyToMany")
            .Select(m => new { M = m, P = m.GetParameters() })
            .Where(x => x.P[0].ParameterType.GetGenericArguments()[0].GetGenericArguments()[1] == typeof(object))
            .FirstOrDefault().M.MakeGenericMethod(elementType);

        dynamic m2m = method.Invoke(table, new object[] { MakePropertyAccessExpression(p)});
        m2m.Table(tableName).ParentKeyColumn("Parent" + typeof(T).Name + "Id").ChildKeyColumn("Child" + elementType.Name + "Id");
    }
    protected Expression<Func<T, object>> MakePropertyAccessExpression(PropertyInfo property)
    {
        var param = Expression.Parameter(property.DeclaringType, "x");
        var ma = Expression.MakeMemberAccess(param, property);
        return Expression.Lambda<Func<T, object>>(ma, param);
    }
}

How do I get the same result with automappings?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文