如何使用Entity Framework Code First和DDD创建数据库?

时间:2013-09-17 12:29:31

标签: entity-framework ef-code-first domain-driven-design

使用实体框架实现域驱动设计我使用Julie Lerman在TechEd North America 2013(http://channel9.msdn.com/Events/TechEd/NorthAmerica/2013/DEV-B336#fbid=4tnuPF6L-Jc)上提出的方法。此方法使用EF实体类作为域类。对于不同的有界上下文,域实体类具有不同的属性,甚至可以有不同的名称,尽管它们将数据存储在同一个表中。示例是“客户服务”有界上下文中的客户,实际上是“客户”,但在“发货”有界上下文中,他是仅具有客户属性子集的“收件人”。对于每个有界上下文,存在不同的EF上下文,其仅包括有界上下文所需的实体的DbSet。通过重写OnModelCreating,我们甚至可以排除与有界上下文无关的引用实体。使用POCO可以很容易地实现这一部分。

问题是使用Code First时的数据库创建。如果我们让Code First为每个不同的EF上下文创建数据库,我们最终会得到几个数据库。如果我们在EF上下文的构造函数中定义数据库名称,则使用第一个使用的EF上下文创建数据库,并在使用第二个EF上下文时丢失InvalidOperationException(说模型已更改)(缺少实体,缺少属性等)。如果使用的EF上下文在使用之前使用其他EF上下文的实体/成员,我们可能会使用迁移来更新数据库。但是,这肯定会与正常使用迁移相混淆,并且无法正常运行。 作为临时解决方案,我仅使用单独的EF上下文来创建数据库。这意味着我必须再次为此目的实现所有EF实体。另一个问题是我必须在应用程序启动时创建此EF上下文的实例,以确保创建数据库并(如果需要)迁移。

我确信还有其他解决方案。所以,请(朱莉)告诉我们如何。

2 个答案:

答案 0 :(得分:2)

从本质上讲,我相信您必须拥有一个主要上下文,其中定义了所有“表”并且还驱动迁移。此上下文是用于创建数据库的内容。

所有后续的“有界”上下文在其构造函数中都有Database.SetInitializer(null),以防止它们篡改数据库模式。

此外,您的主上下文和“有界”上下文都应该从具有连接字符串和此类集的抽象基本上下文类继承。

当您启动应用程序时,您可以简单地尝试实例化主上下文并确保它已迁移到最新版本。但是在稍后的实际应用程序中,您只使用仅限实现主上下文子集的“有界”上下文。

我意识到你已经在部分或整体上做了一些,但我认为这是要走的路。

答案 1 :(得分:1)

我同意Julie Lermans的建议。有一段时间她一直建议采用这种方法。 你不需要EF6这样做。虽然使用EF6可以更容易管理。

根据需要声明任意数量的上下文。 关键是在上下文创建期间数据库的初始化设置。 基本模式如下

例如

// when you wish to migrate
Database.SetInitializer(new MigrateDatabaseToLatestVersion<YOURCONTEXT, YourContentConfiguration>());
var connie = new YOURCONTEXT(.....);

//When you wish to access but NOT change the DB with a small context.
Database.SetInitializer(new ContextInitializerNone<BoundedMiniDbContext>());
var connie = new BoundedMiniDbContext(.....);

其中yourContent继承自DbContext,YourContextConfig继承DbMigrationsConfiguration