c# - Entity Framework Core 2.1 - 拥有类型和嵌套值对象
问题描述
我正在学习 DDD,我目前正在学习的教程是使用 NHibernate 实现的,但由于我缺乏这方面的经验,我决定使用 EF Core 2.1 完成课程。
但是,我目前有点坚持以下几点:我有三个类Customer
,它们是一个实体和两个值对象(CustomerStatus
以及它内部的值对象ExpirationDate
) - 像这样:
public class Customer : Entity
{
//... constructor, other properties and behavior omitted for the sake of simplicity
public CustomerStatus Status { get; set; }
}
public class CustomerStatus : ValueObject<CustomerStatus>
{
// customer status is enum containing two values (Regular,Advanced)
public CustomerStatusType Type { get; }
public ExpirationDate ExpirationDate { get; }
}
public class ExpirationDate : ValueObject<ExpirationDate>
{
//... constructor, other properties and behavior omitted for the sake of simplicity
public DateTime? Date { get; private set; }
}
当我尝试在我的内部执行以下操作时DbContext
:
modelBuilder.Entity<Customer>(table =>
{
table.OwnsOne(x => x.Status,
name =>
{
name.Property(x => x.ExpirationDate.Date).HasColumnName("StatusExpirationDate");
name.Property(x => x.Type).HasColumnName("Status");
});
});
我收到以下错误:
表达式 'x => x.ExpirationDate.Date' 不是有效的属性表达式。该表达式应该表示一个简单的属性访问:'t => t.MyProperty'。
参数名称:propertyAccessExpression'
除此之外,我还尝试过执行以下操作:
table.OwnsOne(x => x.Status.ExpirationDate,
name =>
{
name.Property(x => x.Date).HasColumnName("StatusExpirationDate");
});
table.OwnsOne(x => x.Status,
name =>
{
name.Property(x => x.Type).HasColumnName("Status");
});
但这也导致:
表达式 'x => x.Status.ExpirationDate' 不是有效的属性表达式。该表达式应该表示一个简单的属性访问:'t => t.MyProperty'。
我也试过:
modelBuilder.Entity<Customer>()
.OwnsOne(p => p.Status,
cb => cb.OwnsOne(c => c.ExpirationDate));
但也没有运气......无论如何,任何帮助将不胜感激,如果有人能解释为什么我的尝试都不起作用,如果可能的话,那将是非常棒的?提前致谢!
更新
在按照 Ivan 的评论中所述进行操作后,我收到了关于CustomerStatus
类构造函数的错误,因此我添加了默认受保护的构造函数。
之后我开始收到错误:
实体类型“CustomerStatus”的字段“k__BackingField”是只读的,因此无法设置。
CustomerStatus
如果有帮助,这是我班级的内部:
public class CustomerStatus : ValueObject<CustomerStatus>
{
public CustomerStatusType Type { get; }
public ExpirationDate ExpirationDate { get; }
public static readonly CustomerStatus Regular =
new CustomerStatus(CustomerStatusType.Regular, ExpirationDate.Infinite);
public bool IsAdvanced => Type == CustomerStatusType.Advanced && !ExpirationDate.IsExpired;
private CustomerStatus(CustomerStatusType type, ExpirationDate expirationDate)
{
Type = type;
ExpirationDate = expirationDate;
}
protected CustomerStatus()
{
}
public static CustomerStatus Create(CustomerStatusType type, ExpirationDate expirationDate)
{
return new CustomerStatus(type, expirationDate);
}
public CustomerStatus Promote()
{
return new CustomerStatus(CustomerStatusType.Advanced, ExpirationDate.Create(DateTime.UtcNow.AddYears(1)).Value);
}
protected override bool EqualsCore(CustomerStatus other)
{
return Type == other.Type && ExpirationDate == other.ExpirationDate;
}
protected override int GetHashCodeCore()
{
return Type.GetHashCode() ^ ExpirationDate.GetHashCode();
}
}
更新
所需要的只是在类内添加私有设置器Type
和ExpirationDate
属性,CustomerStatus
并结合伊万的回答,它就像一个魅力。非常感谢!
解决方案
您的尝试不起作用,因为拥有的类型只能通过其所有者实体进行配置,更具体地说,通过OwnsOne
方法返回的它们自己的构建器或作为所有者实体构建器方法的参数的Action<T>
参数提供。OwnsOne
所以配置应该是这样的(注意嵌套OwnsOne
):
modelBuilder.Entity<Customer>(customer =>
{
customer.OwnsOne(e => e.Status, status =>
{
status.Property(e => e.Type).HasColumnName("Status");
status.OwnsOne(e => e.ExpirationDate, expirationDate =>
{
expirationDate.Property(e => e.Date).HasColumnName("StatusExpirationDate");
});
});
});
推荐阅读
- c++ - 在 Windows vscode 的 c++ 程序中无法接收输入(cin)
- django - 具有基于类的视图的 django 分页
- python - 如何在matplotlib中以百万为单位缩放直方图y轴
- r - 从带有列表的列中检索单个值
- javascript - 如何在 React-Native 中创建字典应用程序?
- solr - solr 8.4.1 搜索忽略重复的短语
- html - 菜单弹出,悬停时,在css上
- azure-devops - Azure Pipeline 阶段的变量
- sql-server - 根据多个文本框上显示的文本选择列
- android - 升级 android build:gradle:4.0.0 后在 proguard 中显示错误