使用EF代码优先迁移将新的必填字段添加到表之一

Amm*_*han 8 c# entity-framework ef-code-first ef-migrations

我正在使用EF代码优先迁移。我已经有很多关于生产Db的数据,我想介绍一个不可为空的字段。怎么可能呢?

当前它抛出一个错误:

The column cannot contain null values. [ Column name = Test,Table name = 'MyTable'] 
Run Code Online (Sandbox Code Playgroud)

bin*_*nki 8

我通常使用的策略是先引入新列作为可选列,填充它,然后使其成为必需。实际上,您可以通过对每个步骤进行单独的迁移来进行单独的迁移步骤,或者手动更改自动生成的迁移代码以使更改全部在一次迁移中进行。我将描述如何使用单个迁移。

如果添加新[Required]字段,则自动生成的迁移可能看起来像这样,如果其中dbo.MyTable包含数据,则将失败:

public override void Up()
{            
    AddColumn("dbo.MyTable", "MyColumn", c => c.String(nullable: false));
}

public override void Down()
{
    DropColumn("dbo.MyTable", "MyColumn");
}
Run Code Online (Sandbox Code Playgroud)

您可以编辑迁移,以最初将列添加为可选列,对其进行预填充,然后将其标记为必需。您可以Sql()用来执行预填充。根据您的数据,您可能希望基于其他列甚至另一个表来计算新列的值,并且可以通过编写适当的SQL来实现。

public override void Up()
{
    AddColumn("dbo.MyTable", "MyColumn", c => c.String());
    Sql("UPDATE dbo.MyTable SET MyColumn = COALESCE(SomeOtherColumn, 'My Fallback Value')");
    AlterColumn("dbo.MyTable", "MyColumn", c => c.String(nullable: false));
}

public override void Down()
{
    DropColumn("dbo.MyTable", "MyColumn");
}
Run Code Online (Sandbox Code Playgroud)

如果您只想在所有行上设置相同的值,则可以在删除之前跳过此Sql()步骤,并根据需要添加带有的列。这使您可以在迁移期间将所有行设置为相同的值,同时仍然需要您在添加新行时手动指定该值。据信这种方法不适用于旧版本的EF。我不确定需要哪个版本的EF,但它至少应该在现代版本上可用(?6.2):defaultValuedefaultValue

public override void Up()
{
    AddColumn("dbo.MyTable", "MyColumn", c => c.String(defaultValue: "Some Value", nullable: false));
    AlterColumn("dbo.MyTable", "MyColumn", c => c.String(nullable: false));
}

public override void Down()
{
    DropColumn("dbo.MyTable", "MyColumn");
}
Run Code Online (Sandbox Code Playgroud)