使用EF Code First Migration向表中的一个表添加新的必填字段

时间:2014-12-03 14:23:46

标签: c# entity-framework ef-code-first ef-migrations

我正在使用EF Code First Migration。我已经有很多关于生产Db的数据,我想介绍一个不可空的字段。怎么可能?

目前它会抛出错误:

The column cannot contain null values. [ Column name = Test,Table name = 'MyTable'] 

1 个答案:

答案 0 :(得分:5)

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

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

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

public override void Down()
{
    DropColumn("dbo.MyTable", "MyColumn");
}

您可以编辑迁移,以最初将列添加为可选列,对其进行预填充,然后将其标记为必需。您可以使用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");
}

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

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");
}