首页 > 解决方案 > 来自不同来源的 EF Core 实体继承,没有鉴别器

问题描述

我正在尝试使用 EF Core 配置下面显示的两个实体:

用户:

public class User
{
    public int UserId { get; set; }
    public string Title { get; set; }
    public string Name { get; set; }
    public string Surname { get; set; }
    [Required]
    public string Email { get; set; }

    [NotMapped]
    public string FriendlyIdentifier
    {
        get
        {
            var name = $"{Name} {Surname}";
            if (!string.IsNullOrWhiteSpace(name))
                return name;
            else if (!string.IsNullOrWhiteSpace(Email))
                return Email;
            else
                return UserId.ToString();
        }
    }

    public ICollection<Booking> Bookings { get; set; }
}

用户详情:

public class UserDetails : User
{
    public DateTime? LatestBookingDate { get; set; }
}

这样做的原因是我希望能够使用他们的最新预订日期来拉取用户,而不必拉取所有预订,然后在代码中手动确定最新的预订日期。为此,我创建了一个选择最新预订日期的 SQL 视图,并且我想绑定UserDetails到该视图。

我有以下 EF 设置:

modelBuilder.Entity<User>(builder =>
{
    builder.ToTable("userdetails");
    builder.HasKey(x => x.UserId);
    builder.HasIndex(x => new { x.Name, x.Surname, x.Email });
});

modelBuilder.Query<UserDetails>()
    .ToView("view_userdetails");

但这在尝试选择UserDetails实体时会出现以下错误:

'不能将'User'设置为'UserDetails'的基本类型。继承层次结构不能包含实体类型和查询类型的混合。

我知道HasDiscriminator这里描述了一个扩展:https ://docs.microsoft.com/en-us/ef/core/modeling/relational/inheritance但我并没有真正的鉴别器,因为UserDetails它只是一个扩展User.

我有什么办法让这个工作吗?我真的不想复制Userinto的属性,UserDetails因为当发生任何变化时,需要维护两个实体。我想保留遗产。

标签: c#asp.netentity-frameworkasp.net-coreentity-framework-core

解决方案


我想保留遗产。

你不能(至少不是现在这样)。异常消息清楚地表明了这一点 - 它说“不能包含混合”并且没有为您提供建议。

您可以做的是将您当前的User类成员移动到一个新的基础非实体非查询类型类,并让User实体和UserDetails查询类型都继承自它,例如:

public abstract class UserInfo
{
    public int UserId { get; set; }
    public string Title { get; set; }
    public string Name { get; set; }
    public string Surname { get; set; }
    [Required]
    public string Email { get; set; }

    [NotMapped]
    public string FriendlyIdentifier
    {
        get
        {
            var name = $"{Name} {Surname}";
            if (!string.IsNullOrWhiteSpace(name))
                return name;
            else if (!string.IsNullOrWhiteSpace(Email))
                return Email;
            else
                return UserId.ToString();
        }
    }

    public ICollection<Booking> Bookings { get; set; }
}

public class User : UserInfo { }

public class UserDetails : UserInfo
{
    public DateTime? LatestBookingDate { get; set; }
}

这将允许您在一个地方维护通用模型。缺点是UserDetails无法在期望的地方使用User,并且必须转换(尽管由于所有常见属性都相同,因此对于 AutoMapper 来说应该是微不足道的“自动”任务)。

请注意,此解决方案要求您在查询类型中Ignore使用Bookings导航属性:

modelBuilder.Query<UserDetails>(builder =>
{ 
    builder.ToView("view_userdetails");
    builder.Ignore(e => e.Bookings);
});

或将其移至User实体。


推荐阅读