我正在使用EF Code First Migration。我已经有很多关于生产Db的数据,我想介绍一个不可空的字段。怎么可能?
目前它会抛出错误:
The column cannot contain null values. [ Column name = Test,Table name = 'MyTable']
答案 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");
}