5

我希望有人能够帮助我,因为我似乎完全陷入了困境。

对于我们公司即将开展的项目,我们希望将 Entity Framework 5 与代码优先方法一起使用。我玩了一会儿,每次尝试将 EF 与我们现有的库一起使用时,我都失败了,因为 EF 似乎严重依赖现有的 app.config。

在我们公司,我们有一个内部数据库库,它允许我们连接到各种数据源和数据库技术,利用数据库提供商的 MEF(托管扩展框架)的优势。我只需传递一些数据库设置,例如主机(或文件)、目录、用户凭据和数据库提供程序名称,库会查找适当的插件并返回自定义连接字符串或 IDbConnection。我们希望将此库与 EF 一起使用,因为它使我们能够灵活地选择我们使用的数据库,并在运行时更改数据库。

所以。我看到一个典型的 DbContext 对象在构造函数中没有参数。它会自动在 app.config 中查找适当的连接字符串。我们不喜欢这样的事情,所以我更改了默认构造函数以获取传递给 DbContext 基类的 DbConnection 对象。没有成交。

当代码优先模型更改时会出现问题。EF 会自动注意到这一点并查找迁移类/配置。但是:典型的迁移类需要上下文的默认无参数构造函数!太遗憾了!

因此,我们使用 IDbContextFactory 接口构建自己的迁移类。但同样,这个 IDbContextFactory 似乎也需要一个无参数的构造函数,否则我无法添加迁移或更新数据库。

此外,我制作了自己的数据迁移配置器,在其中传递上下文以及目标数据库。问题就在这里:无论我尝试什么,它都找不到任何迁移类。

我完全被卡住了,因为似乎使用 EF 的唯一方法是将连接字符串保存在 app.config 中。这很愚蠢,因为我们需要在运行时更改数据库连接,而 app.config 对默认用户是只读的!

如何解决这个问题?

4

2 回答 2

3

答案在这里提供

https://stackoverflow.com/a/15919627/941240

诀窍是稍微修改默认MigrateDatabaseToLatestVersion初始化程序,以便:

  • 数据库总是被初始化...
  • ...使用当前上下文中的连接字符串

仍然会创建一个新的DbMigrator数据上下文,但会根据初始化程序从您的上下文中复制连接字符串。我什至能够缩短代码。

就这样:

public class MasterDetailContext : DbContext
{
    public DbSet<Detail> Detail { get; set; }
    public DbSet<Master> Master { get; set; }

    // this one is used by DbMigrator - I am NOT going to use it in my code
    public MasterDetailContext()
    {
        Database.Initialize( true );
    }

    // rather - I am going to use this, I want dynamic connection strings
    public MasterDetailContext( string ConnectionString ) : base( ConnectionString )
    {
        Database.SetInitializer( new CustomInitializer() );
        Database.Initialize( true );
    }

    protected override void  OnModelCreating(DbModelBuilder modelBuilder)
    {
        modelBuilder.Conventions.Remove<PluralizingTableNameConvention>();
    }
}

public class CustomInitializer : IDatabaseInitializer<MasterDetailContext>
{

    #region IDatabaseInitializer<MasterDetailContext> Members

    // fix the problem with MigrateDatabaseToLatestVersion 
    // by copying the connection string FROM the context
    public void InitializeDatabase( MasterDetailContext context )
    {            
        Configuration cfg = new Configuration(); // migration configuration class
        cfg.TargetDatabase = new DbConnectionInfo( context.Database.Connection.ConnectionString, "System.Data.SqlClient" );

        DbMigrator dbMigrator = new DbMigrator( cfg );
        // this will call the parameterless constructor of the datacontext
        // but the connection string from above will be then set on in
        dbMigrator.Update();             
    }

    #endregion
}

客户端代码:

    static void Main( string[] args )
    {

        using ( MasterDetailContext ctx = new MasterDetailContext( @"Database=ConsoleApplication801;Server=.\SQL2012;Integrated Security=true" ) )
        {
        }

        using ( MasterDetailContext ctx = new MasterDetailContext( @"Database=ConsoleApplication802;Server=.\SQL2012;Integrated Security=true" ) )
        {
        }
    }

运行这将导致根据迁移配置创建和迁移两个数据库。

于 2013-04-29T11:32:16.637 回答
0

它需要一个无参数的构造函数才能调用它。您可以做的是DbConntectionFactory在空构造函数中提供默认值,例如:

public DbContext()
{
    IDbContextFactory defaultFactory; //initialize your default here
    DbContext(defaultFactory);
}

public DbContext(IDbContextFactory factory)
{
}
于 2013-04-25T09:33:06.120 回答