原文链接: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. Android开发-mac上使用三星S3做真机调试

    之前一直未使用真机进行Android开发,为准备明天的培训,拿出淘汰下来的s3准备环境,竟然发现无法连接mac,度娘一番找到答案,如下:mac 系统开发android,真机调试解决方案(无数的坑之后吐 ...

  2. java io读书笔记(7) Closing Output Streams

    输出完毕后,需要close这个stream,从而使操作系统释放相关的资源.举例: public void close( ) throws IOException 并不是所有的stream都需要clos ...

  3. Object-c 语法 - NSObject常用方法和反射

    NSObject常用方法 - (BOOL)isKindOfClass:(Class)aClass //判断是否为aClass或者aClass的子类的实例,aClass可以通过[类名 class]获取 ...

  4. 比较X与Y的大小,绝对精准!!!!!!

    代码! #include<stdio.h> int max(int a,int b) { int x; x=a+b; return x; } int main() { int i,n,t; ...

  5. django解决跨域请求的问题

    跨域请求可以用jsonp来解决,不过今天我发现一个很好用的包:django-cors-headers 只需要简单地配置一下就可 被请求方的setting.py中的配置如下: INSTALLED_APP ...

  6. 在android源码环境下写上层应用的一个初步解决方法

    在android源码环境下编写上层应用,一直以来我都觉得很麻烦.因为如果单纯将应用导入eclipse,那么framework层一些定制的API无法自动提示和补全,使用起来不太方便:如果将整个andro ...

  7. C语言库函数大全及应用实例二

    原文:C语言库函数大全及应用实例二                                              [编程资料]C语言库函数大全及应用实例二 函数名: bioskey 功 能 ...

  8. 201521123063 《Java程序设计》 第10周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结异常与多线程相关内容. 2. 书面作业 本次PTA作业题集异常.多线程 finally 题目4-2 1.1 截图你的提交结果(出现学 ...

  9. GIT入门笔记(12)- 删除文件、提交删除和恢复删除

    在Git中,删除也是一个修改操作,我们实战一下, 1.先添加add一个新文件test.txt到Git并且提交commit到本地版本库: $ git add test.txt$ git commit - ...

  10. Dynamic CRM 2015学习笔记(6)没有足够的权限 - 您没有访问这些记录的权限。请联系 Microsoft Dynamics CRM 管理员

    我们经常遇到下面这种问题:没有足够的权限 - 您没有访问这些记录的权限.请联系 Microsoft Dynamics CRM 管理员.  下面将详细介绍下如何解决这种问题:进不了CRM系统:进了CRM ...