Я использую FluentNHibernate (Automapping) для отображения, NHibernate 3.2 для доступа к данным и SchemaExport для создания моей базы данных.
У меня есть класс Principal
, который является базовым классом для User
и Usergroup
,Principal
имеет свойство CommonThing
типа CommonThing
.CommonThing
имеет 2 набора: ManagedUsers
и ManagedUsergroups
.
Теперь создается столбец CommonThingId
для Principals
-таблицы (ОК), Users
-таблицы (НЕПРАВИЛЬНО), Usergroups
-table (WRONG).
Как заставить FluentNHibernate на генерировать столбец только в Principals
-table , а не в подклассах таблиц?
Редактировать: КлассыИ сопоставления Принципал:
public abstract class Principal : Entity
{
...
public virtual CommonThing CommonThing
{
get
{
return _commonThing;
}
set
{
if (_commonThing == value)
return;
_commonThing = value;
if (_commonThing == null)
return;
if (this is Usergroup)
_commonThing.AddUsergroup(this as Usergroup);
else if (this is User)
_commonThing.AddUser(this as User);
}
}
...
}
Пользователь:
public partial class User : Principal
{
...
}
Группа пользователей:
public partial class Usergroup : Principal
{
...
}
CommonThing:
public class CommonThing : Entity
{
...
public virtual IEnumerable<User> ManagedUsers { get { return _managedUsers; } set { _managedUsers = (Iesi.Collections.Generic.ISet<User>)value; } }
public virtual IEnumerable<Usergroup> ManagedUsergroups { get { return _managedUsergroups; } set { _managedUsergroups = (Iesi.Collections.Generic.ISet<Usergroup>)value; } }
...
}
Условные обозначения:
public class ReferenceConvention : IReferenceConvention
{
public void Apply(IManyToOneInstance instance)
{
var keyName = string.Format(CultureInfo.InvariantCulture, "FK_MtO_{0}_in_{1}_{2}",
instance.Property.PropertyType.Name,
instance.EntityType.Name,
instance.Name);
instance.ForeignKey(keyName);
instance.LazyLoad();
instance.Cascade.SaveUpdate();
instance.Column(instance.Property.PropertyType.Name + "Id");
instance.Access.CamelCaseField(CamelCasePrefix.Underscore);
}
}
public class ForeignKeyConvention : FluentNHibernate.Conventions.ForeignKeyConvention
{
protected override string GetKeyName(Member property, Type type)
{
if (property == null)
return type.Name + "Id";
return property.Name + "Id";
}
}
public class HasManyConvention : IHasManyConvention
{
public void Apply(IOneToManyCollectionInstance instance)
{
var keyName = string.Format(CultureInfo.InvariantCulture, "FK_OtM_{0}_{1}2{2}",
instance.Member.ReflectedType.Name,
instance.Member.Name,
instance.EntityType.Name);
instance.Key.ForeignKey(keyName);
if(instance.Key.Columns.Count() != 0)
instance.Inverse();
instance.Cascade.SaveUpdate();
instance.Cache.ReadWrite();
instance.Cache.IncludeAll();
instance.Access.CamelCaseField(CamelCasePrefix.Underscore);
}
}
public class JoinedSubclassConvention : IJoinedSubclassConvention
{
public void Apply(IJoinedSubclassInstance instance)
{
instance.Table("" + Inflector.Net.Inflector.Pluralize(instance.Type.Name));
instance.Key.Column("Id");
instance.DynamicInsert();
instance.DynamicUpdate();
instance.LazyLoad();
}
}
Основное отображение:
public class PrincipalMapping : IAutoMappingOverride<Principal>
{
public void Override(AutoMapping<Principal> mapping)
{
...
mapping.References(x => x.CommonThing)
.LazyLoad()
.Nullable()
.Access.CamelCaseField(Prefix.Underscore)
.Cascade.None();
;
mapping.JoinedSubClass<User>("Id");
mapping.JoinedSubClass<Usergroup>("Id");
...
}
}
Общее отображение:
public class CommonThingMapping : IAutoMappingOverride<CommonThing>
{
public void Override(AutoMapping<CommonThing> mapping)
{
...
mapping.HasMany(x => x.ManagedUsers)
.AsSet()
.ExtraLazyLoad()
;
mapping.HasMany(x => x.ManagedUsergroups)
.ExtraLazyLoad()
.AsSet()
;
...
}
}
Lg
warappa