在Entity Framework中使用我自己的域模型对象和.edmx

时间:2013-02-14 21:19:02

标签: entity-framework domain-driven-design poco

我有一个域模型体系结构,其中我的域/业务对象是基于问题域创建的,并且独立于物理数据模型或持久性结构的任何知识。到目前为止,我已经走上正轨,因为它完全可以接受通常情况下,域模型和数据模型之间存在阻抗不匹配。 DBA创建了数据库以获取所需的数据 ,但它并未封装应用程序整个域模型或设计。

结果 - 我有自己的域模型对象集。但是,所有需要持久存储的字段存在于我的域模型中的某个位置,但不一定是我自动生成的.edmx POCO实体拥有它们的形状。所以我拥有所有数据,它与完美形状完全不同,就像创建自动生成的POCO实体的表一样。

我看过一些关于此主题的帖子,例如converting POCO entity to business entityEntity Framework 4 with Existing Domain Model,它们的语句如下:

  

“在实体数据模型中创建具有相同名称的实体   你的域名类。实体属性也应该相同   域类“

中的名称和类型

什么!没办法,为什么我必须将我的域模型重新整理到在数据库中的数据模型/表结构之后建模完全的POCO?例如 - 在我有5个给定属性的情况下,2可能在类'A'中,3在类'B'中,而自动生成的POCO类在其自己的类'A'中具有全部5。

这就是重点,我想要分离我的对象模型和数据模型,但仍然使用像EF 5.0这样的ORM来映射它们。我不想在数据模型中创建和形成这样命名的类和属性。

现在我在EF 5.0中的.edmx正在为我生成POCO类,但我的问题是如何解散这些并将所有内容重新连接到我的域对象,其中包含所有这些数据但只是在不同的形状?

顺便说一下,使用Code First方法提出的任何解决方案都不是一个选项,所以不提供此功能。我需要一些指导或教程(最好)使用EF5(如果可能,因为EF4示例总是从ObjectContext继承POCO),并将我自己的业务对象连接到.edmx。

感谢任何帮助或指导,谢谢!

3 个答案:

答案 0 :(得分:1)

这听起来就像是Entity Framework的用例。我在这里做了一些假设。首先,当你发表这个声明时:

“我有一个域模型体系结构,其中我的域/业务对象是基于问题域创建的,并且独立于物理数据模型或持久性结构的任何知识。”

你的意思是这个域是在EF设计师中创建的?但是你说:

“但是,所有需要持久保存的字段确实存在于我的域模型中的某个位置,但不一定是我自动生成的.edmx POCO实体拥有它们的形状。”

这听起来像我的第一个假设是不正确的。

接下来,你首先解雇代码?如果您的域模型/业务对象是基于代码的,并且您希望将它们持久保存到关系数据库,那么这就是代码的用例。你有代码,现在你需要创建你的DbContext并将其映射到你的物理模型。

然而你不理会......所以有些想法:

如果您有基于代码的业务对象的域模型,并且您有一个用于其他事情的EDMX,我认为您希望创建一个存储库层,使用自动映射器或手动投影来查询您的实体并返回你的业务对象。

如果你有一个基于代码的业务对象的域模型,并且你有一个EDMX,除了持久化业务对象之外没有用于其他事情,我会说你需要在EDMX中表达你的域,然后将它映射到你现有的数据库。这实际上是ORM的用例。拥有两个域模型并从一个模型映射到另一个模型,其中一个模型与您的域匹配,一个匹配您的数据库,这就是添加一个额外的不需要的管道层。

上面的方法在EF的说法中被称为“模型优先”。虽然大部分文章只是从模型中生成数据库,但有几篇文章都是关于它的。您不会执行该步骤,而是将实体映射到现有数据库。

这方面的基本步骤是“从数据库更新”而不选择任何数据库对象(或将创建实体)。或者,您可以在设计器中使用退出的.edmx(这听起来像你有)并修改实体以匹配您的业务领域。或者只是删除EDMX模型中的所有实体,根据需要创建实体,然后将它们全部映射。

这是我在使用EF Designer引入模型商店时所做的一个jing(唯一的方法是允许它生成实体),然后通过单击NO删除允许商店信息保留的实体当它询问您是否要删除表信息时。

http://screencast.com/t/8eiPg2kcp

我没有将POCO生成器添加到此,但如果我这样做,它将在设计器中生成实体作为POCO类。

答案 1 :(得分:0)

上面引用的声明并不是说您重写域对象以匹配您的pocos,而是建议您更新edmx以匹配您的域模型。

在您的示例中,您可以在edmx中创建一个实体,该实体映射两个表中的所有5个属性,EF将管理从单个生成的Poco到表的映射。

当然这意味着您有重复的域对象和pocos,这意味着您在与EF交互时必须手动将域对象转换为pocos,

或者您可以将域数据对象定义为接口,并提供pocos的部分实现,这些实现基本上将每个poco标识为域对象的具体实现。

可能还有其他一些方法可以修饰这个特殊的猫,但我认为你不能提供预定义的c#对象以供在edmx中使用。

答案 2 :(得分:0)

一种方法可能是选择ViewModel(适合您的特定业务逻辑)并自动将上下文中的一些数据映射到它https://stackoverflow.com/a/8588843/201648。这使用AutoMapper将实体属性从EF上下文映射到ViewModel。这不会为你做任何事情,但它可能会让生活变得更轻松。如果您对自动发生的方式不满意,可以将AutoMapper配置为略有不同(请参阅投影) - https://github.com/AutoMapper/AutoMapper/wiki/Projection

您可能已经知道这一点,但也可以使用t4 - http://visualstudiogallery.msdn.microsoft.com/72a60b14-1581-4b9b-89f2-846072eff19d从您的EDMX自动生成POCO。如果您定义模板以生成分部类,则可以使用另一个具有该POCO的自定义属性的分部类。这样,您可以自动填充大多数属性,但是具有其他自定义属性,您可以使用上下文/存储库中的自定义规则填充这些属性。这需要很多单调来生成这些,然后您可以专注于使用上述技术重塑数据。

如果您对两者都非常不满意,您可以始终映射存储过程以自动获取您想要的确切字段名称,而无需填充。这当然会影响您使用数据的方式,但我之前已经完成了优化目的/已经存在的程序完全符合我的要求。见http://msdn.microsoft.com/en-us/data/gg699321.aspx

相关问题