4

一年多后,从我关于 SO(从子实体的属性中过滤基本实体)的第一个问题开始,我遇到了类似的问题。

我有一个抽象基类型

public abstract class Base{
}

我有一些继承自此基本类型的子实体

public class Child1 : Base{
    public virtual NavigationProperty NavigationProperty {get; set; }
    public int NavigationPropertyId {get; set}
}

public class Child2 : Base{
    public virtual NavigationProperty NavigationProperty {get; set; }
}

子实体都具有 NavigationProperty 属性。NavigationProperty 类就像

 public class NavigationProperty{
    public virtual ICollection<Child1> Child1s {get; set;}
    public virtual Child2 Child2s {get; set;}
 }

Child2 和 NavigationProperty 之间存在一对一的映射关系;Child1 和 NavigationProperty 之间的一对多关系。为了使这些映射起作用,我正在使用 TPT。我的第一个问题是,我可以搬家吗

  public NavigationProperty NavigationProperty {get; set; }

到基类?

我正在尝试一整天,但没有任何成功。如果不可能,我至少可以从基本类型访问 NavigationProperty。在所有孩子都有这个属性之后,我尝试了类似的东西

 public abstract class Base{
      public abstract NavigationProperty NavigationProperty {get; set; }
 }
 ....
 public abstract class Child2{
      public override NavigationProperty NavigationProperty {get; set; }
 }

但是实体框架给出了以下错误。

 Sequence contains more than one matching element 

我可以使用类似的东西

   public abstract class Base{
      public abstract NavigationProperty GetNavigationProperty();
   }

  public abstract class Child2{
      public override NavigationProperty NavigationProperty {get; set; }
      public override NavigationProperty GetNavigationProperty(){
           return NavigationProperty;
      }
 }

但我不想介绍这些额外的方法。他们有什么方法可以更优雅地实现这一目标吗?

编辑:

我忘了提到我已经尝试过放置[NotMapped]属性。我猜 EF[NotMapped]属性也是继承的,所以子属性也没有映射。

我不希望 Linq-to-Entites 能够工作。我不希望能够查询具有导航属性的基本实体。我只是想摆脱 GetNavigationProperty 和 SetNavigationProperty 方法。因此,当我尝试从基类访问 NavigationProperty 时,它应该被加载到内存中,仅此而已。然而,经过一周的努力,我认为这是不可能的。

4

3 回答 3

1

编辑 x1更新代码以避免属性名称的文字字符串

一点点反思似乎可以完成这项工作。班级设置;

public class NavigationProperty
{
    public NavigationProperty(string name)
    {
        Name = name;
    }

    public string Name { get; set; }
}

public abstract class Base
{
    public NavigationProperty NavigationProperty
    {
        get
        {
            string propertyName = MethodBase.GetCurrentMethod().Name.Replace("get_", string.Empty);
            PropertyInfo property = this.GetType().GetProperty(propertyName);
            if (property != null)
            {
                nav = (NavigationProperty)property.GetValue(this, new object[] { });
            }

            return nav;
        }
        set
        {
            string propertyName = MethodBase.GetCurrentMethod().Name.Replace("set_", string.Empty);
            PropertyInfo property = this.GetType().GetProperty(propertyName);
            if (property != null)
            {
                property.SetValue(this, value, new object[] { });
            }
        }
    }
}

public class Child1 : Base {
    public NavigationProperty NavigationProperty { get; set; }
    public int NavigationPropertyId { get; set; }
}

public class Child2 : Base{
    public NavigationProperty NavigationProperty { get; set; }
}

在你的代码中;

Child1 c1 = new Child1() { NavigationProperty = new NavigationProperty("child1Value") };
Child2 c2 = new Child2() { NavigationProperty = new NavigationProperty("child2Value") };
Base somebase = c1;
NavigationProperty childNav = somebase.NavigationProperty;
// childNav.Name now contains "child1Value"

这符合您的要求吗?它比使用方法有点笨拙,abstract但至少意味着您不必重构每个子类

于 2013-03-07T17:52:35.680 回答
1

我可以移动吗

public NavigationProperty NavigationProperty {get; set; }

到基类?

不,因为实体中的逆属性NavigationProperty指的是Child1and Child2,而不是Base。导航属性始终必须是声明类型的属性,并且不能移动到继承链中的基类型。

对于第二个问题,您可以尝试从映射中排除抽象导航属性:

public abstract class Base {
    [NotMapped]
    public abstract NavigationProperty NavigationProperty {get; set; }
}

(或modelBuilder.Entity<Base>().Ignore(b => b.NavigationProperty);使用 Fluent API)。

您将无法Base.NavigationProperty在任何查询中使用,因为您不能将未映射的属性与 LINQ-to-Entities 一起使用。

于 2013-03-04T19:05:55.803 回答
1

在 Entity Framework 中找不到针对您的场景的支持,您可以尝试以下操作:

public interface IHasNavigationProperty {
    NavigationProperty NavigationProperty { get; }
}

public class Child1 : Base, IHasNavigationProperty {
    public NavigationProperty NavigationProperty { get; set; }
}

public class Base {
    public void AMethodThatDoesStuff() {
        if (this is IHasNavigationProperty) {
            var navigationProperty = ((IHasNavigationProperty)this).NavigationProperty;

            /* do stuff with NavigationProperty */
        }
    }
}
于 2013-03-08T01:46:29.050 回答