原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/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(50).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 Server. 代表本机

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

connectionString=“Server=.;Database=AppsDB;Trusted_Connection=True” 也可以

控制台中:

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 = 0;
//转换成功,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(50).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 = 0;
//转换成功,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类:修改一下代码:(必须把DbMigrationsConfiguration的参数改成自己程序里的DbContext的namespace+实际类名)

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

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(50).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 = 0;
//转换成功,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);
}
}
}

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

看下数据库:

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

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

  1. 2.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. EF Code First教程-03 数据库迁移Migrator

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

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

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

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

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

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

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

  8. 【EF Code First】Migrations数据库迁移

    1,打开工具->NuGet程序管理器->程序包管理器控制台 默认项目中要选择  数据访问上下文类  所在的项目 我的DB是在命名空间CodeFirst.UI下的所以选择CodeFirst. ...

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

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

随机推荐

  1. 08-为数组和arguments

    <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8&quo ...

  2. [BZOJ3611] [Heoi2014]大工程(DP + 虚树)

    传送门 $dp[i][0]$表示节点i到子树中的所有点的距离之和 $dp[i][1]$表示节点i到子树中最近距离的点的距离 $dp[i][2]$表示节点i到子树中最远距离的点的距离 建好虚树后dp即可 ...

  3. 刷题总结——学姐的逛街计划(vijos1891费用流)

    题目: doc 最近太忙了, 每天都有课. 这不怕, doc 可以请假不去上课.偏偏学校又有规定, 任意连续 n 天中, 不得请假超过 k 天. doc 很忧伤, 因为他还要陪学姐去逛街呢. 后来, ...

  4. 【2018.10.10】[HNOI2008] GT考试(bzoj1009)

    10pts: 暴力枚举字符串,Hash判是否出现.(真会有人写么) 时间复杂度$O(10^n*n)$. 40pts: 学过OI的人都会写的dp 如果这道题的40pts($n\le 250000$)设成 ...

  5. ubuntu mysql安装及需要其他主机连服务器mysql时的设置(error:10061)

    说明: 一个朋友在使用ubuntu-server 16.04安装mysql,设置远程访问的时候出现了问题,请我帮忙.但是,我也没有使用过ubuntu安装mysql,于是乎搜索了很多技术文件,比着葫芦画 ...

  6. JSP表单提交中文乱码

    简要笔记:由于jsp默认表单提交编码方式是:ISO-8859-1,而我们需要的是utf-8或者是gbk码,故需要转化. 具体方法是:在表单处理文件中,将获取到的变量进行转换. String userN ...

  7. 洛谷 P1522 牛的旅行

    题目描述 农民 John的农场里有很多牧区.有的路径连接一些特定的牧区.一片所有连通的牧区称为一个牧场.但是就目前而言,你能看到至少有两个牧区通过任何路径都不连通.这样,Farmer John就有多个 ...

  8. 「CodePlus 2018 4 月赛」最短路

    $n \leq 100000$,$m \leq 500000$的有向图,两点之间还可以以$a \ \ xor \ \ b$的代价从$a$到$b$,问$s$到$t$的最短路. 被自己蠢哭QAQ 首先两个 ...

  9. Laravel 数据库操作之Eloquent ORM模型

    //模型中的相关代码 namespace App; use Illuminate\Database\Eloquent\Model; class Student extends Model{ //默认对 ...

  10. Day 7 Linux之系统监控、硬盘分区等

    Linux之系统监控.硬盘分区等 系统监控 系统监视和进程控制工具—top和free 1) 掌握top命令的功能:top命令是Linux下常用的性能分析工具,能够实时显示系统中各个进程的资源占用状况, ...