原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/code-first-migrations-with-entity-framework/

或者:http://www.codeproject.com/Articles/801545/Code-First-Migrations-With-Entity-Framework

系列目录:

前面的文章中,学习了EF 中的几种关系,一对一,一对多,多对多。但是相信大家肯定会有疑问:

1.我难道每次都要创建数据库么?

2.我怎么样从已经存在的表中,添加字段和移除字段呢?

3.当我向表中,添加字段或者移除字段,我怎么来避免丢失数据呢?

4.当数据库发生改变的时候,我怎么获取到数据库的脚本呢?

不用着急,这篇文章,我会向大家一一讲到:

首先,说说我们为什么要使用数据库迁移技术吧,因为我们的实体总是变动地很频繁,在上篇文章中,我们使用了数据库初始化策略来做,也就是每次当数据库不存在的时候,就创建数据库【类似还有几种初始化策略】,然而,当你的实体改变的时候,在使用这个策略,EF就会报错。而数据库迁移技术就可以帮到我们,我们不用每次都创建数据库。并且数据库迁移技术还可以为我们设置初始化的数据。

先看看项目结构吧:

我们需要建2个类库项目,还有一个控制台的程序:

Student类:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace EF.Core
{
public class Student
{
public int ID { get; set; } public string Name { get; set; } public int Age { get; set; } public string Sex { get; set; }
}
}

StudentMap类:

using EF.Core;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Ef.Data
{
public class StudentMap:EntityTypeConfiguration<Student>
{
public StudentMap()
{
this.HasKey(s => s.ID);
this.Property(s => s.ID).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
this.Property(s => s.Name).HasColumnType("nvarchar").HasMaxLength().IsRequired();
this.Property(s => s.Sex).HasColumnType("nvarchar").IsRequired();
this.Property(s => s.Age).IsRequired(); this.ToTable("Students");
}
}
}

EF上下文类:

using System;
using System.Collections.Generic;
using System.Data.Entity;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Ef.Data
{
public class EFDbContext:DbContext
{
public EFDbContext()
: base("name=DbConnectionString")
{ } protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Configurations.Add(new StudentMap());
//base.OnModelCreating(modelBuilder);
}
}
}

数据库连接字符串:【在EF.Data的配置文件和控制台的项目的配置文件中都要写:】另外注意:这两个项目需要引入EF

<connectionStrings>
<add name="DbConnectionString" connectionString="Server=.;database=StudentEFDB;uid=sa;pwd=Password_1" providerName="System.Data.SqlClient" />
</connectionStrings>

控制台中:

using Ef.Data;
using EF.Core;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace EF.App
{
public class Program
{ static void Main(string[] args)
{
Console.WriteLine("你好,请输入姓名:");
string name = Console.ReadLine().Trim();
Console.WriteLine("请输入年龄:");
int result = ;
//转换成功,result是正确的结果
if (!int.TryParse(Console.ReadLine().Trim(), out result))
{
Console.WriteLine("请输入正确格式的年龄");
return;
}
Console.WriteLine("请输入你的性别:");
string sex = Console.ReadLine(); using(var db=new EFDbContext())
{
Student model = new Student()
{
Name = name,
Sex = sex,
Age = result }; //db.Set<Student>().Add(model);或者下面的
db.Entry(model).State = System.Data.Entity.EntityState.Added;
db.SaveChanges(); }
Console.Write("Success!");
Console.ReadKey(); }
}
}

运行之后,写入数据。

看下数据库中的数据:

好了,这就完成了第一阶段的准备工作:现在我们需要在Student实体中加一个字段Email,项目做一下变动:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace EF.Core
{
public class Student
{
public int ID { get; set; } public string Name { get; set; } public int Age { get; set; } public string Sex { get; set; } public string Email { get; set; }
}
}
using EF.Core;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Ef.Data
{
public class StudentMap:EntityTypeConfiguration<Student>
{
public StudentMap()
{
this.HasKey(s => s.ID);
this.Property(s => s.ID).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
this.Property(s => s.Name).HasColumnType("nvarchar").HasMaxLength().IsRequired();
this.Property(s => s.Sex).HasColumnType("nvarchar").IsRequired();
this.Property(s => s.Age).IsRequired();
this.Property(s => s.Email).IsRequired(); this.ToTable("Students");
}
}
}
using Ef.Data;
using EF.Core;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace EF.App
{
public class Program
{ static void Main(string[] args)
{
Console.WriteLine("你好,请输入姓名:");
string name = Console.ReadLine().Trim();
Console.WriteLine("请输入年龄:");
int result = ;
//转换成功,result是正确的结果
if (!int.TryParse(Console.ReadLine().Trim(), out result))
{
Console.WriteLine("请输入正确格式的年龄");
return;
}
Console.WriteLine("请输入你的性别:");
string sex = Console.ReadLine(); Console.WriteLine("请输入你的Email:");
string email = Console.ReadLine(); using(var db=new EFDbContext())
{
Student model = new Student()
{
Name = name,
Sex = sex,
Age = result,
Email=email }; //db.Set<Student>().Add(model);或者下面的
db.Entry(model).State = System.Data.Entity.EntityState.Added;
db.SaveChanges(); }
Console.Write("Success!");
Console.ReadKey(); }
}
}

运行之后:报错,很正常嘛,这肯定报错,因为我们的实体改了。。

怎么办呢??别着急,我们来启用数据库迁移技术:

在程序包控制台中输入:

Enable-Migrations

接着按回车键,在项目中就会生成Migration文件夹,自己去看看里面有啥东东吧。

然后打开Configuration类:修改一下代码:

接着在程序包管理器控制台中输入:

Update-Database -Verbose

好了,现在运行一下项目吧:

看到了么,到这里就没报错了。运行完之后,我们看下数据库中的数据:

可以看到之前的数据也在,也就是数据没有丢失。是不是很神奇呢???

总结:这两篇文章总结了EF的一些使用,希望你喜欢。

题外篇:这里是使用EF中的数据库迁移技术。我们就可以随时随地的添加删除字段。而不用手动去删除数据库,那样会丢失数据。

现在,我就想不用数据库迁移呢?

我删掉之前生成的Migration文件夹【包括里面的类】,然后修改代码【删除刚才添加的Email字段】:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace EF.Core
{
public class Student
{
public int ID { get; set; } public string Name { get; set; } public int Age { get; set; } public string Sex { get; set; } // public string Email { get; set; }
}
}
using EF.Core;
using System;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity.ModelConfiguration;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Ef.Data
{
public class StudentMap:EntityTypeConfiguration<Student>
{
public StudentMap()
{
this.HasKey(s => s.ID);
this.Property(s => s.ID).HasDatabaseGeneratedOption(DatabaseGeneratedOption.Identity);
this.Property(s => s.Name).HasColumnType("nvarchar").HasMaxLength().IsRequired();
this.Property(s => s.Sex).HasColumnType("nvarchar").IsRequired();
this.Property(s => s.Age).IsRequired();
// this.Property(s => s.Email).IsRequired(); this.ToTable("Students");
}
}
}
using Ef.Data;
using EF.Core;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace EF.App
{
public class Program
{ static void Main(string[] args)
{
Console.WriteLine("你好,请输入姓名:");
string name = Console.ReadLine().Trim();
Console.WriteLine("请输入年龄:");
int result = ;
//转换成功,result是正确的结果
if (!int.TryParse(Console.ReadLine().Trim(), out result))
{
Console.WriteLine("请输入正确格式的年龄");
return;
}
Console.WriteLine("请输入你的性别:");
string sex = Console.ReadLine(); //Console.WriteLine("请输入你的Email:");
//string email = Console.ReadLine(); using(var db=new EFDbContext())
{
Student model = new Student()
{
Name = name,
Sex = sex,
Age = result,
// Email=email }; //db.Set<Student>().Add(model);或者下面的
db.Entry(model).State = System.Data.Entity.EntityState.Added;
db.SaveChanges(); }
Console.Write("Success!");
Console.ReadKey(); }
}
}

运行之后,还肯定会报错的,因为已经没有数据库迁移了:

我们做如下修改:

using System;
using System.Collections.Generic;
using System.Data.Entity;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace Ef.Data
{
public class EFDbContext:DbContext
{
public EFDbContext()
: base("name=DbConnectionString")
{
//把数据库初始化策略设置为null
Database.SetInitializer<EFDbContext>(null);
} protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Configurations.Add(new StudentMap());
//base.OnModelCreating(modelBuilder);
}
}
}

然后运行项目,就不错报错了,这是不使用数据库迁移技术做到的。

看下数据库:

两种方式都可以,不过还是推荐官方的方式,数据库迁移技术。

2.EF中 Code-First 方式的数据库迁移的更多相关文章

  1. EF中 Code-First 方式的数据库迁移

    原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/code-first-migrations-with-entity-framework/ 系列目 ...

  2. EF 中 Code First 的数据迁移以及创建视图

    写在前面: EF 中 Code First 的数据迁移网上有很多资料,我这份并没什么特别.Code First 创建视图网上也有很多资料,但好像很麻烦,而且亲测好像是无效的方法(可能是我太笨,没搞成功 ...

  3. MVC中code first方式开发,数据库的生成与更新

    在使用EF的实际编程中我们经常遇到这样的问题:发现实体结构需要新增加一个字段,或者减少一个字段,急需把实体结构修改,并让数据库更新这种修改.在用Model First或者Database First的 ...

  4. 【EF6学习笔记】(一)Code First 方式生成数据库及初始化数据库实际操作

    本篇参考原文地址: Creating an Entity Framework Data Model 说明:学习笔记参考原文中的流程,为了增加实际操作性,并能够深入理解,部分地方根据实际情况做了一些调整 ...

  5. EF6 学习笔记(一):Code First 方式生成数据库及初始化数据库实际操作

    EF6 学习笔记总目录:ASP.NET MVC5 及 EF6 学习笔记 - (目录整理) 本篇参考原文地址: Creating an Entity Framework Data Model 说明:学习 ...

  6. EF Code First教程-03 数据库迁移Migrator

    要在nuget 程序包管理控制台中输入命令 基本命令 Enable-Migrations   //打开数据库迁移 Add-Migration AddBlogUrl    //新增一个数据库迁移版本   ...

  7. EF架构~codeFirst从初始化到数据库迁移

    一些介绍 CodeFirst是EntityFrameworks的一种开发模式,即代码优先,它以业务代码为主,通过代码来生成数据库,并且加上migration的强大数据表比对功能来生成数据库版本,让程序 ...

  8. 【转】MVC中code first方式开发,数据库的生成与更新(Ef6)

    一,在models文件夹中,建立相应的model文件         这里注意一点,这里建立的class名,就是数据库里表的名字.         在这里面,可以建立表之间的关系. 这里要说明一点的事 ...

  9. EF中创建、使用Oracle数据库的Sequence(序列)功能

    ** 背景 ** 项目中订单号原来的生成规则由日期加随机数组成,后期需求决定将订单号生成规则更改为生成日期加当天当前订单数. 每天的订单数都是从0开始的,每生成一个订单,订单数就应该加1.订单数应该是 ...

随机推荐

  1. redis 学习笔记(2)

    redis-cluster 简介 redis-cluster是一个分布式.容错的redis实现,redis-cluster通过将各个单独的redis实例通过特定的协议连接到一起实现了分布式.集群化的目 ...

  2. jquery.Callbacks的实现

    前言 本人是一个热爱前端的菜鸟,一直喜欢学习js原生,对于jq这种js库,比较喜欢理解他的实现,虽然自己能力有限,水平很低,但是勉勉强强也算是能够懂一点吧,对于jq源码解读系列,博客园里有很多,推荐大 ...

  3. LINQ to SQL Select查询

    1. 查询所有字段 using (NorthwindEntities context = new NorthwindEntities()) { var order = from n in contex ...

  4. HTML 5 应用程序缓存manifest

    什么是应用程序缓存(Application Cache)? HTML5 引入了应用程序缓存,这意味着 web 应用可进行缓存,并可在没有因特网连接时进行访问. 应用程序缓存为应用带来三个优势: 离线浏 ...

  5. SharePoint 2013: A feature with ID has already been installed in this farm

    使用Visual Studio 2013创建一个可视web 部件,当右击项目选择"部署"时报错: "Error occurred in deployment step ' ...

  6. ExecuteOrDelayUntilScriptLoaded 还是 SP.SOD.executeFunc?

    SharePoint 客户端 JS 开发时,要等待 SharePoint 对象都加载完毕再调用自己的方法(myFunction),可以有两种方式: ExecuteOrDelayUntilScriptL ...

  7. 敏捷转型历程 - Sprint4 回顾会

    我: Tech Leader 团队:团队成员分布在两个城市,我所在的城市包括我有4个成员,另外一个城市包括SM有7个成员.另外由于我们的BA离职了,我暂代IT 的PO 职位.PM和我在一个城市,但他不 ...

  8. mono for android 读取网络远程图片

    布局 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android=& ...

  9. 【Java并发编程实战】-----“J.U.C”:CAS操作

    CAS,即Compare and Swap,中文翻译为"比较并交换". 对于JUC包中,CAS理论是实现整个java并发包的基石.从整体来看,concurrent包的实现示意图如下 ...

  10. CSharpGL(35)用ViewPort实现类似3DMax那样的把一个场景渲染到4个视口

    CSharpGL(35)用ViewPort实现类似3DMax那样的把一个场景渲染到4个视口 开始 像下面这样的四个视口的功能是很常用的,所以我花了几天时间在CSharpGL中集成了这个功能. 在CSh ...