EntityFramework Core 学习系列(一)Creating Model
EntityFramework Core 学习系列(一)Creating Model
Getting Started
使用Command Line 来添加 Package
dotnet add package Microsoft.EntityFrameworkCore.SqlServer
使用 -v
可以指定相应包的版本号。
使用dotnet ef
命令
需要在.csproj
文件中包含下面引用
<ItemGroup>
<DotNetCliToolReference Include="Microsoft.EntityFrameworkCore.Tools.DotNet" Version="2.0.0"/>
</ItemGroup>
Creating a Model
Fluent API
在继承至 DbContext
的子类中,重载 OnModelCreating() 方法进行Fluent API 的配置。
public class MyDbContext : DbContext
{
//...
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<TEntity>()
.Property(b => b.Url)
.IsRequired();
//...
}
//...
}
Data Annotation
或者可以使用数据注解直接在实体中进行配置:
public class Blogs
{
public string BlogName { get; set; }
[Required]
public string Url { get; set; }
}
关于配置的顺序规范,
Fluent API > Data Annotations > Conventions
Include & Exclude
有下列三种情况类或实体会被包含:
- By convention, types that are exposed in
DbSet
properties on your context are included in your model. - Types that are mentioned in the
OnModelCreating
method are also included. - Any types that are found by recursively exploring the navigation properties of discovered types are also included in the model.
你可以通过 Data Annotation 或者 Fluent API 来 排除包含,示例如下:
//Data Annotations Example Below:
public class Blogs
{
public int BlogId { get; set;}
public BlogAuthor BlogAuthors { get; set;}
}
[NotMapped]
public class BlogAuthor
{
public string FirstName { get; set;}
//...
}
//Fluent API Example Below:
public class MyDbContext : DbContext
{
public DbSet<Blog> Blogs { get; set;}
protected override OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Ignore<BlogAuthor>();
}
}
public class Blogs
{
public int BlogId { get; set;}
public BlogAuthor BlogAuthors { get; set;}
}
public class BlogAuthor
{
public string FirstName { get; set;}
//...
}
当然除了可以排出/包含类之外, 还可以自己配置相应的属性如下:
//Data Annotations Example Below:
public class Blogs
{
public int BlogId { get; set;}
public BlogAuthor BlogAuthors { get; set;}
[NotMapped]
public DateTime BlogAddedTime { get; set;}
}
//Fluent API Example Below:
public class MyDbContext : DbContext
{
public DbSet<Blog> Blogs { get; set;}
protected override OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Blog>()
.Ignore(b => b.BlogAddedTime);
}
}
public class Blogs
{
public int BlogId { get; set;}
public BlogAuthor BlogAuthors { get; set;}
public DateTime BlogAddedTime { get; set;}
}
Key
关于EF Core 中的Key, 按照规范,如果一个属性命名为Id 或者 以Id结尾的都会被配置成该实体的主键。比如下面的这个:
public class TestClass
{
public int Id { get; set; }
//or
public int MyId { get; set; }
}
或者你可以按照 Data Annotation 或者 Fluent API 来进行配置:
// Data Annotations
public class Car
{
[Key]
public string CarLicense { get; set; }
public string CarFrameCode { get; set;}
}
//Fluent API
//...
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder.Entity<Car>()
.HasKey(c => c.CarLicense);
}
//...
// Multiple Properties to the key ,复合主键的配置只能用Fluent API
modelBuilder.Entity<Car>()
.HasKey(c => new { c.CarLicense, c.CarFrameCode });
Generated Values
中文应该是叫值的自动生成
吧,我也不清楚。EF Core
上 有三种 Value Generation Pattern。分别是
No Value Generation
No value generation means that you will always supply a valid value to be saved to the database. This valid value must be assigned to new entities before they are added to the context.
值不自动生成,每个属性的值都需要指定,添加。这个我理解的意思应该是你保存到数据库里面的值,每个必须是有效的,并且需要指定。
Value Generated on Add
Value generated on add means that a value is generated for new entities.
Depending on the database provider being used, values may be generated client side by EF or in the database. If the value is generated by the database, then EF may assign a temporary value when you add the entity to the context. This temporary value will then be replaced by the database generated value during
SaveChanges()
.If you add an entity to the context that has a value assigned to the property, then EF will attempt to insert that value rather than generating a new one. A property is considered to have a value assigned if it is not assigned the CLR default value (
null
forstring
,0
forint
,Guid.Empty
forGuid
, etc.).属性的值在 添加到数据库时自动添加。
Value Generated on Add or Update
Value generated on add or update means that a new value is generated every time the record is saved (insert or update).
Like
value generated on add
, if you specify a value for the property on a newly added instance of an entity, that value will be inserted rather than a value being generated. It is also possible to set an explicit value when updating.属性的值在 添加到数据库或者更新时自动添加。
当当看着上面这三个,我其实并不知道这三个到底是什么意思,接下来会用例子来演示一下这些具体意思,以及使用规范。
By convention, primary keys that are of an integer or GUID data type will be setup to have values generated on add. All other properties will be setup with no value generation.(当主键为 Interger 或者 GUID类型事,该主键的值会自动生成。)
下面用实例来模拟一下:
我们新建一个实体 Blogs 如下:
//Entity Blog
public class Blog
{
public int BlogId { get; set; }
public string Url { get; set; }
}
// Program.cs
using (var db = new BookDbContext())
{
if (!db.Blogs.Any())
{
var blog = new Blog
{
Url = "Https://q.cnblogs.com"
};
db.Blogs.AddRange(blog);
db.SaveChanges();
}
}
直接用在Program 用using 来演示效果,由于我们之前说到的主键为 Integer 类型的会自动生成值,所以数据库中的值大家可想而知,就是下面这个
下面是用Data Annotation(数据注解)来演示的,也可以用Fluent API:
No Value Generation
如果我们不想让它自动生成的话呢,也有办法。向下面这样:
public class Blog
{
[DatabaseGenerated(DatabaseGeneratedOption.None)]
public int BlogId { get; set; }
public string Url { get; set; }
}
我在 BlogId 上加上 DatabaseGeneratedOption.None 之后,我们再重新运行上面的程序,发现数据库的值如下所示:
为什么是 0 的原因呢,其实上面已经解释过了:就是下面这句话
A property is considered to have a value assigned if it is not assigned the CLR default value (
null
forstring
,0
forint
,Guid.Empty
forGuid
, etc.). CLR 的默认值。
Fluent API 版本
modelBuilder.Entity<Blog>()
.Property(b => b.BlogId)
.ValueGeneratedNever();
Value Generated on Add
比如我们还想在 Blog 实体里面加一个 更新时间 UpdateTime 属性
public class Blog
{
public int BlogId { get; set; }
public string Url { get; set; }
[DataGenerated(DatabaseGeneratedOption.Identity)]
public DateTime UpdateTime { get; set;}
}
当我们配置成上面这样,然后直接 dotnet run 时,发现程序报错了。
Unhandled Exception: Microsoft.EntityFrameworkCore.DbUpdateException: An error occurred while updating the entries. See the inner exception for details. ---> Microsoft.Data.Sqlite.SqliteException: SQLite Error 19: 'NOT NULL constraint failed: Blogs.UpdateTime'.
在Setting Explicit Value 中在OnModelCreating 中配置,使其自动生成,但是本地我使用SQLite 时无法自动生成,报错。具体使用如下
modelBuilder.Entity<Blog>()
.Property(p => p.UpdateTime)
.HasDefaultValueSql("CONVERT(date, GETDATE())");
Fluent API 版本
modelBuilder.Entity<Blog>()
.Property(b => b.UpdateTime)
.ValueGeneratedOnAdd();
Value generated on add or update (Data Annotations)
public class Blog
{
public int BlogId { get; set; }
public string Url { get; set; }
[DatabaseGenerated(DatabaseGeneratedOption.Computed)]
public DateTime LastUpdated { get; set; }
}
Fulent API 版本
modelBuilder.Entity<Blog>()
.Property(b => b.LastUpdated)
.ValueGeneratedOnAddOrUpdate();
下面记录一下 dotnet ef 命令的使用
dotnet ef migrations add name
dotnet ef database update
dotnet ef migrations remove
dotnet ef database drop
EntityFramework Core 学习系列(一)Creating Model的更多相关文章
- ASP.NET Core学习系列
.NET Core ASP.NET Core ASP.NET Core学习之一 入门简介 ASP.NET Core学习之二 菜鸟踩坑 ASP.NET Core学习之三 NLog日志 ASP.NET C ...
- EntityFramework Core 学习笔记 —— 创建模型
原文地址:https://docs.efproject.net/en/latest/modeling/index.html 前言: EntityFramework 使用一系列的约定来从我们的实体类细节 ...
- EntityFramework Core 学习扫盲
0. 写在前面 1. 建立运行环境 2. 添加实体和映射数据库 1. 准备工作 2. Data Annotations 3. Fluent Api 3. 包含和排除实体类型 1. Data Annot ...
- Net core学习系列(一)——Net Core介绍
一.什么是Net Core .NET Core是适用于 windows.linux 和 macos 操作系统的免费.开源托管的计算机软件框架,是微软开发的第一个官方版本,具有跨平台 (Windows. ...
- EntityFramework Core 学习笔记 —— 添加主键约束
原文地址:https://docs.efproject.net/en/latest/modeling/keys.html Keys (primary) Key 是每个实体例的主要唯一标识.EF Cor ...
- 【.Net Core 学习系列】-- EF Core 实践(Code First)
一.开发环境: VS2015, .Net Core 1.0.0-preview2-003156 二解决方案: 新建项目: File --> New --> Project --> ...
- 【.Net Core 学习系列】-- EF Core实践(DB First)
一.开发环境: VS2015, .Net Core 1.0.0-preview2-003156 二.准备数据: CREATE DATABASE [Blogging]; GO USE [Blogging ...
- Net core学习系列(八)——Net Core日志
一.简介# 日志组件,作为程序员使用频率最高的组件,给程序员开发调试程序提供了必要的信息.ASP.NET Core中内置了一个通用日志接口ILogger,并实现了多种内置的日志提供器,例如 Conso ...
- Net core学习系列(四)——Net Core项目执行流程
"跨平台"后的ASP.Net Core是如何接收并处理请求的呢? 它的运行和处理机制和之前有什么不同?本章从"宏观"到"微观"地看一下它的结 ...
随机推荐
- Jquery判断单选框是否选中和获取选中的值
第一种:利用选中值判断选中 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http ...
- Java基础学习笔记二十二 网络编程
络通信协议 通过计算机网络可以使多台计算机实现连接,位于同一个网络中的计算机在进行连接和通信时需要遵守一定的规则,这就好比在道路中行驶的汽车一定要遵守交通规则一样.在计算机网络中,这些连接和通信的规则 ...
- Java虚拟机之性能监控
一.jstat:虚拟机统计信息监控工具监视虚拟机各种运行状态 图中,S0.S1(Survivor0.Survivor1)代表两个Survivor区,其中一个值为57.60%.另一个为0.E(Eden) ...
- 关于如何学习C语言
2016级计算机专业的C语言分为两个学期,第一学期是C语言(基础),第二学期是C语言(高级),在第一学期主要学习的内容是基本的数据类型,分支结构和循环结构,一维和二维数组,字符数组,函数.通过这学期独 ...
- bug终结者 团队作业第六、七周
bug终结者 团队作业第六.七周 作业要求:团队作业第六.七周 博客编辑:20162322 朱娅霖 一.修改<需求规格说明书> <需求规格说明书>2.0版(即初稿) <需 ...
- 敏捷冲刺每日报告——Day3
1.情况简述 Alpha阶段第一次Scrum Meeting 敏捷开发起止时间 2017.10.27 00:00 -- 2017.10.28 00:00 讨论时间地点 2017.10.27晚9:30, ...
- 【iOS】swift 保持代码优美的10个方法
这篇Swift风格指南与你看到的其他的指南有所不同,此篇指南主要焦点集中在打印和Web展示的可读写上.我们创建此篇风格指南的目的,是为了让我们的图书.教程以及初学者套件中的代码保持优美和一致,即使我们 ...
- MapReduce工作机制——Word Count实例(一)
MapReduce工作机制--Word Count实例(一) MapReduce的思想是分布式计算,也就是分而治之,并行计算提高速度. 编程思想 首先,要将数据抽象为键值对的形式,map函数输入键值对 ...
- MHA 安装与简单使用
MHA 在过去几年一直用的比较火,特别是在在传统复制的那个年代.至从有了GTID好像我们也可以把MHA给忘记了,但是很多企业现在还是在用的比较多.每个公司的MHA玩法也不太一样,但是本质都是差不多了. ...
- 如何深入系统的学习一门编程语言——python自学笔记
前言 最早接触python的时候,他并没有现在这么火,我也没把他太当回事,那时候我对python的印象就是给运维人员使用的一门很古老的语言,显然随着tensorflow(以下简称tf)的兴起,pyth ...