建立初始模型和数据库数据库
在开始使用迁移(Migrations)以前,咱们须要一个 Project 和一个 Code First Model, 对于本文将使用典型的 Blog 和 Post 模型服务器
public class Blog { public int BlogId { get; set; } public string Name { get; set; } }
public class BlogContext : DbContext { public DbSet<Blog> Blogs { get; set; } }
更改 Program.cs 以调用ide
static void Main(string[] args) { using (var db = new BlogContext()) { db.Blogs.Add(new Blog { Name = "Another Blog" }); db.SaveChanges(); foreach (var blog in db.Blogs) { Console.WriteLine(blog.Name); } } Console.WriteLine("Press any key to exit..."); Console.ReadKey(); }
运行查看结果函数
发现如上错误"CREATE DATABSE permission denied in databse 'master'"post
咱们在 BlogContext 上的无参构造函数上添加诊断代码并设置调试断点测试
System.Diagnostics.Debug.Write(Database.Connection.ConnectionString);
再次运行ui
咱们注意到 Data Scource 居然是 .\\SQLEXPRESS 而不是咱们想要的 localDB , 这是由于:spa
知道了缘由咱们就好解决了:调试
<connectionStrings> <add name="BlogContext" connectionString="Data Source=(LocalDb)\v11.0;Initial Catalog=BlogContext;Integrated Security=SSPI;" providerName="System.Data.SqlClient"/> </connectionStrings>
再次运行就好了,让咱们看一下后台生成的数据库code
启用迁移
咱们对模型 Blog 作一些更改:增长一个 Url 属性
public string Url { get; set; }
咱们此时再次运行程序,发现以下错误
'InvalidOperationException' was unhandled. The model backing the 'BlogContext' context has changed since the database was created. Consider using Code First Migrations to update the database ( http://go.microsoft.com/fwlink/?LinkId=238269)
正如错误消息提示的那样,是时候使用 Code First Migrations,第一步是运行以下的命令:
这个命令将在项目下建立文件夹 Migrations
多个实体锁定同一数据库
当使用 EF6 以前的版本时,只会有一个 Code First Model 被用来生成/管理数据库的 Schema, 这将致使每一个数据库只会有一张 __MigrationsHistory 表,从而没法辨别实体与模型的对应关系。
从 EF6 开始,Configuration 类将会包含一个 ContextKey 属性,它将做为每个 Code First Model 的惟一标识符, __MigrationsHistory 表中一个相应地的列容许来自多个模型(multiple models)的实体共享表(entries),默认状况下这个属性被设置成 context 的彻底限定名。
生成、运行迁移
Code First Migrations 有两个你须要熟悉的命令:
咱们须要脚手架(scaffold 直译)一个迁移,以上面的 Url 属性为例,命令 Add-Migration 容许咱们对迁移命名,咱们姑且称之为 AddBlogUrl
namespace MigrationsDemo.Migrations { using System; using System.Data.Entity.Migrations; public partial class AddBlogUrl : DbMigration { public override void Up() { AddColumn("dbo.Blogs", "Url", c => c.String()); } public override void Down() { DropColumn("dbo.Blogs", "Url"); } } }
咱们如今能够对这个迁移进行编辑或者增长,但彷佛看起来还不错,那咱们就直接用 Update-Database 来应用到数据库吧
定制化迁移
到目前为止咱们生成并运行了一个迁移,可是没有对迁移作任何更改,下面咱们将尝试作一些更改:在类 Bolg 上增长一属性 Rating
public int Rating { get; set; }
新建 Post
public class Post { public int PostId { get; set; } [MaxLength(200)] public string Title { get; set; } public string Content { get; set; } public int BlogId { get; set; } public Blog Blog { get; set; } }
在 Blog 中添加 Post 的集合
public virtual ICollection<Post> Posts { get; set; }
在 Package Manager Console 中运行命令 Add-Migration AddPostClass
生成的迁移以下
接下来咱们对迁移作些更改:
更改后的代码以下
namespace MigrationsDemo.Migrations { using System; using System.Data.Entity.Migrations; public partial class AddPostClass : DbMigration { public override void Up() { CreateTable( "dbo.Posts", c => new { PostId = c.Int(nullable: false, identity: true), Title = c.String(maxLength: 200), Content = c.String(), BlogId = c.Int(nullable: false), }) .PrimaryKey(t => t.PostId) .ForeignKey("dbo.Blogs", t => t.BlogId, cascadeDelete: true) .Index(t => t.BlogId) .Index(p => p.Title, unique: true); AddColumn("dbo.Blogs", "Rating", c => c.Int(nullable: false, defaultValue: 3)); } public override void Down() { DropIndex("dbo.Posts", new[] { "Title" }); DropForeignKey("dbo.Posts", "BlogId", "dbo.Blogs"); DropIndex("dbo.Posts", new[] { "BlogId" }); DropColumn("dbo.Blogs", "Rating"); DropTable("dbo.Posts"); } } }
在 Package Manager Console 中运行命令 Update-Database –Verbose
数据移动 / 定制SQL
迄今为止,迁移都没有更改或移动数据,如今让咱们看一下须要移动数据的例子。虽然没有对数据移动的原生支持,可是咱们能够随意运行 SQL 脚本。
让咱们在 Post 中增长一个属性 Abstract, 稍后咱们使用列 Content 的开头来填充此列(数据库已有记录)
public string Abstract { get; set; }
namespace MigrationsDemo.Migrations { using System; using System.Data.Entity.Migrations; public partial class AddPostAbstract : DbMigration { public override void Up() { AddColumn("dbo.Posts", "Abstract", c => c.String()); Sql("UPDATE dbo.Posts SET Abstract = LEFT(Content, 100) WHERE Abstract IS NULL"); } public override void Down() { DropColumn("dbo.Posts", "Abstract"); } } }
在 Package Manager Console 中运行命令 Update-Database –Verbose
迁移至指定版本(包括后退)
迄今为止,咱们老是升级至最新迁移,然而某些时候咱们须要升级/降级至指定版本,例如咱们想迁移数据库至运行 AddBlogUrl 迁移以后的状态,此时咱们就可使用 –TargetMigration 来降级到这个版本
在 Package Manager Console 中运行命令 Update-Database –TargetMigration: AddBlogUrl
这个命令将会运行 AddBlogAbstract and AddPostClass 的 Down 命令
若是你想回滚一切至空数据库,可使用命令 Update-Database –TargetMigration: $InitialDatabase
获得SQL脚本
若是其它开发人员也但愿在他们本身的机器上拥有这些更改,他们只需在咱们 check in 代码至 source control 的时候作一次同步便可,一旦他们拥有了这些迁移,只需运行命令 Update-Database 就能够把这些更改应用于本地。可是若是咱们想把这些更改推送至测试服务器或生产服务器,咱们也许须要一份 SQL 脚本提供给 DBA
产生的 SQL 脚本以下
产生幂等脚本(EF6+)
从 EF6 开始,若是你使用 –SourceMigration $InitialDatabase, 产生的脚本将是幂等的,幂等脚本意味着不管数据库当前处于什么版本/状态,都能升级至最新版本或指定版本(指定 –TargetMigration),生成的脚本包括检查表 __MigrationsHistory 的逻辑以及只更新以前从未更新的
在应用程序启动时自动升级(MigrateDatabaseToLatestVersion初始化器)
当你发布部署应用程序的时候,可能但愿当程序启动的时候它自动更新数据库(更新应用任何未更新的迁移),你能够经过注册 MigrateDatabaseToLatestVersion 数据库初始化器来实现这一点,数据库初始化器只包含一些逻辑检查用于确保数据库被正确设置,这个逻辑检查将会在AppDomain 的 context 第一次被使用的时候执行。
当咱们建立一个初始化器的实例时,须要指定 context type(BlogContext)以及 migrations configuration (Configuration)- 这个迁移配置类是在咱们启用迁移时生成的 Migrations 目录下增长的
using System; using System.Collections.Generic; using System.Data.Entity; using System.Linq; using System.Text; using System.Threading.Tasks; using MigrationsDemo.Migrations; namespace MigrationsDemo { class Program { static void Main(string[] args) { Database.SetInitializer(new MigrateDatabaseToLatestVersion<BlogContext, Configuration>()); using (var db = new BlogContext()) { db.Blogs.Add(new Blog { Name = "Another Blog " }); db.SaveChanges(); foreach (var blog in db.Blogs) { Console.WriteLine(blog.Name); } } Console.WriteLine("Press any key to exit..."); Console.ReadKey(); } } }