jjzjj

c# - DropCreateDatabaseIfModelChanges EF6 导致 System.InvalidOperationException : The model backing the context has changed

coder 2024-05-21 原文

迁移到 Entity Framework 6 后,我在构建服务器上执行单元测试时遇到错误。

我正在使用 DropCreateDatabaseIfModelChanges 初始化程序。当我将其更改为 MigrateDatabaseToLatestVersion 时一切正常,但我想坚持使用之前的初始化程序。

我得到的错误是:

System.InvalidOperationException: System.InvalidOperationException: The model backing the 'AppContext' context has changed since the database was created. Consider using Code First Migrations to update the database (http://go.microsoft.com/fwlink/?LinkId=238269)..

这是正确的,它发生了变化,但是使用 DropCreateDatabaseIfModelChanges 初始化器,它应该被重新创建。有什么想法吗?

EF 在 App.config 中配置。这是相关部分:

<connectionStrings>
    <add name="AppContext" connectionString="Data Source=(localdb)\v11.0;Initial Catalog=my.app.unittest;Integrated Security=True" providerName="System.Data.SqlClient" />
</connectionStrings>
<entityFramework>
    <defaultConnectionFactory type="System.Data.Entity.Infrastructure.LocalDbConnectionFactory, EntityFramework">
        <parameters>
            <parameter value="v11.0" />
        </parameters>
    </defaultConnectionFactory>
    <contexts>
        <context type="my.app.core.Data.AppContext, my.app.core">
            <databaseInitializer type="System.Data.Entity.DropCreateDatabaseIfModelChanges`1[[my.app.core.Data.AppContext, my.app.core]], EntityFramework" />
        </context>
    </contexts>
    <providers>
        <provider invariantName="System.Data.SqlClient" type="System.Data.Entity.SqlServer.SqlProviderServices, EntityFramework.SqlServer" />
    </providers>
</entityFramework>

最佳答案

嗯,看起来 EF 6.0 引入了一个新规则:

“如果 DbContext 使用初始化程序并且配置了迁移,则在构建模型时抛出异常”。

直到并包括 EF 6 RC,这并没有被强制执行。烦人的部分是“迁移已配置”是由 DbMigrationsConfiguration 的实现定义的。似乎没有办法在测试中以编程方式禁用迁移 - 如果您实现了

我以与 Sebastian Piu 非常相似的方式解决它 - 我不得不从我的测试中删除 Configuration 类,但我不能只是删除它,因为我们正在为我们的主要项目使用迁移。啊!

这是我之前的代码:

public class MyDbContext : DbDContext, IMyDbContext
{
  public IDbSet<Users> Users {get; set;}
  public IDbSet<Widgets> Widgets {get; set;}
}

// Migrations are considered configured for MyDbContext because this class implementation exists.
internal sealed class Configuration : DbMigrationsConfiguration<MyDbContext>
{
  public Configuration()
  {
    AutomaticMigrationsEnabled = false;
  }
}

// Declaring (and elsewhere registering) this DB initializer of type MyDbContext - but a DbMigrationsConfiguration already exists for that type.
public class TestDatabaseInitializer : DropCreateDatabaseAlways<MyDbContext>
{
    protected override void Seed(MyDbContext context) { }
}

我在测试代码中初始化 DbContext 时遇到了 System.InvalidOperationException。由于该应用程序不使用任何 Initializer,因此运行该应用程序不会像以前那样出现问题。这只会破坏我的测试。

解决方案(感觉更像是 EF 中缺少的东西的变通方法)是将 Initializer 和 DbMigrationsConfiguration 分段,以便在运行时环境中只看到一个。我希望我的测试使用 Initializer,我希望我的应用程序使用 DbMigrationsConfiguration。如果 DbContext 有一个接口(interface),这可以做得更干净,但遗憾的是它只实现了 IObjectContextAdapter。

首先我制作了我的 DbContext 摘要:

public abstract class MyDbContextBase : DbContext, IMyDbContext
{
      public IDbSet<Users> Users {get; set;}
      public IDbSet<Widgets> Widgets {get; set;}
}

然后我派生了2个类:

public class MyDbContext : MyDbContextBase
{
  public MyDbContext(string connectionStringOrName, IDatabaseInitializer<MyDbContext> dbInitializer) 
    : base(connectionStringOrName)
  {
  }
}

public class MyTestDbContext : MyDbContextBase
{
  public MyTestDbContext(string connectionStringOrName, IDatabaseInitializer<MyDbContext> dbInitializer) 
    : base(connectionStringOrName)
  {
    Database.SetInitializer(dbInitializer);
  }
}

MyDbContext 和 MyTestDbContext 都是 IMyDbContext,因此您现有的依赖注入(inject)设置无需更改即可正常工作。我只测试了 Spring.NET。

我的 DbMigrationsConfiguration 实现了测试未使用的派生类型:

internal sealed class Configuration : DbMigrationsConfiguration<MyDbContext>
{
  public Configuration()
  {
    AutomaticMigrationsEnabled = false;
  }
}

最后,初始化器的类型被移动到派生的测试类类型:

public class TestDatabaseInitializer : DropCreateDatabaseAlways<MyTestDbContext>
{
    protected override void Seed(MyTestDbContext context) { }
}

我可以确认我的测试正在通过并且我的应用程序(和迁移)仍在像以前一样工作。

关于c# - DropCreateDatabaseIfModelChanges EF6 导致 System.InvalidOperationException : The model backing the context has changed,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19430502/

有关c# - DropCreateDatabaseIfModelChanges EF6 导致 System.InvalidOperationException : The model backing the context has changed的更多相关文章

随机推荐