EntityFramework 6.x和EntityFramework Core插入数据探讨
前言
一直以来对EF和EF Core都是单独分开来写,从未以比较的形式来讲解,如果您既用过EF 6.x也用过用EF Core是否有了解过EF和EF Core在插入数据时和返回主键有何异同呢?本篇博客是坐在电脑旁本打算写写EF 6.x插入数据注意的问题,心想何不比较二者呢?我也是在探索中(边敲代码边写博客中),下面我们来看看。
EF 6.x和EF Core插入数据异同
using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine;
var customer = new Customer()
{
Email = "2752154844@qq.com",
Name = "Jeffcky",
Orders = new List<Order>()
{
new Order()
{
Code = "",
CreatedTime = DateTime.Now,
ModifiedTime = DateTime.Now,
Quantity = ,
Price =
}
}
};
}
上述Customer和Order为一对多关系,Order实体中有Customer实体的外键,上述我们同时给Customer和Order赋了值,所以当我们插入Customer的同时Order表中也插入了数据,此时Order的CustomerId是Customer的主键,我们根本不需要为Order中的CustomerId显式赋值,这一点毋庸置疑。我想说的是如果两个表没有很强的关联关系,怎么说呢,换言之两个表没有配置所谓的关系又或许我们没有配置关系,一个表中列需要用到另外一个表的主键,那么这种的情况下,我们会以怎样的方式插入数据呢?实践是检验真理的唯一标准,下面我们来试试。
public class TestA
{
public int Id { get; set; }
public string Other { get; set; }
} public class TestB
{
public int Id { get; set; }
public int TestAId { get; set; }
public string Other { get; set; }
}
protected override void OnModelCreating(DbModelBuilder modelBuilder)
{
modelBuilder.Entity<TestA>()
.ToTable("TestAs")
.HasKey(p => p.Id)
.Property(p => p.Other); modelBuilder.Entity<TestB>()
.ToTable("TestBs")
.HasKey(p => p.Id)
.Property(p => p.Other);
}
上述我们给出TestA和TestB,TestA和TestB没有任何关系,但是我们在插入TestB数据时需要得到Test的主键,那我们下面就进行如下数据添加。
using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine;
var testA = new TestA() { Other = "other" };
ctx.TestAs.Add(testA);
ctx.SaveChanges();
var testB = new TestB() { Other = "other", TestAId = testA.Id };
ctx.TestBs.Add(testB);
ctx.SaveChanges();
}

此时我们看到提交后数据最终能够保存到数据库中,反观上述提交代码,我们首先是提交了TestA保存到数据库后然后拿到TestA的主键,然后再是提交TestB并保存到数据库中。那我们有没有考虑是否直接一次性提交呢,注释TestA提交,如下:
using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine;
var testA = new TestA() { Other = "other" };
ctx.TestAs.Add(testA);
//ctx.SaveChanges();
var testB = new TestB() { Other = "other", TestAId = testA.Id };
ctx.TestBs.Add(testB);
ctx.SaveChanges();
}

WOW不行啊,下面我们来看看在EF Core中实现是不是可以,试试就知道了,别猜测。
modelBuilder.Entity<TestA>(e =>
{
e.ToTable("TestAs");
e.HasKey(p => p.Id);
e.Property(p => p.Other);
}); modelBuilder.Entity<TestB>(e =>
{
e.ToTable("TestBs");
e.HasKey(p => p.Id);
e.Property(p => p.Other);
});
using (var context = new EFCoreDbContext())
{
var testA = new TestA() { Other = "other" };
context.TestAs.Add(testA);
//ctx.SaveChanges();
var testB = new TestB() { Other = "other", TestAId = testA.Id };
context.TestBs.Add(testB);
context.SaveChanges();
}

如果分两次提交那么无论是在EF 6.x还是EF Core中都是一样没有任何不同(在EF Core中没有测试也不用测试)。如果是一次性提交,此时在EF 6.x中的TestB中的TestAId为插入的是0,而EF Core中的TestB中的TestAId为-2147482647即INT类型最小值,至少找到了不同所在。Jeff自问自答的模式要来了,是不是就这样结束了呢?上述我们对TestA和TestB两个实体未配置任何关系,我们经过测试证明一次性提交并未达到我们预期,要是我们依然在不配置关系的前提下给出导航属性然后一次性提交呢,如下:
public class TestA
{
public int Id { get; set; }
public string Other { get; set; }
} public class TestB
{
public int Id { get; set; }
public int TestAId { get; set; }
public string Other { get; set; }
public TestA TestA { get; set; }
}
接下来我们在EF Core控制台再次运行上述代码看看,您思考下会不会将TestA中的主键添加进去呢。

如果您在EF 6.x中同样添加上述导航属性也是好使的,我就不测试了,那到此我们得出结论:若两个实体未显式配置任何关系但一个表需要得到另外一个表的主键,无论是在EF 6.x还是在EF Core中进行一次性提交不好使,只是在EF 6.x中一个表需要得到另外一个表的主键为0,而在EF Core中却是INT类型最小值,若我们显式配置了导航属性,那么无论是在EF 6.x还是EF Core中一次性提交可达到我们预期。
比较EF 6.x和EF Core插入数据返回主键
如果是主键为INT类型,默认情况无论是EF 6.x还是EF Core都将自动映射配置为自增长,要是我们显式配置了主键,那么对于EF 6.x和EF Core会有何不同呢?我们首先看看EF 6.x,如下(我们清除之前已提交数据):
using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine; var testA = new TestA() { Id = 1, Other = "other" };
ctx.TestAs.Add(testA);
ctx.SaveChanges();
}

从上述我们提交三次看出,压根不叼我们设置的值,那么我们看看生成的SQL语句是怎样的呢,如下图:

这下我们明白了此时通过scope_identity返回为当前会话和当前作用域中的TestA表生成的最新标识值。接下来我们来看看EF Core。
using (var context = new EFCoreDbContext())
{
var testA = new TestA() { Id = , Other = "other" };
context.TestAs.Add(testA);
context.SaveChanges();
}
当我们进行第二次提交后将抛出异常,如下:

我们同样看看在EF Core中生成的SQL是怎样的。

原来如此没有返回当前会话自增长值,同时我们知道不能显式插入值,那就是关闭了IDENTITY_Insert。所以我们得出结论:在以INT作为主键且自增长时,在EF 6.x中能够显式给出值,而在EF Core中不能显式给定值即关闭了IDENTITY_INSERT不能显式插入主键值。
总结
本节我们详细叙述了EF 6.x和EF Core中插入数据和返回主键的不同之处,虽然作用不大,可以作为了解,最近比较累,可能会停更一小段时间,好好休息一下,书出版了会及时告知同行关注者。
EntityFramework 6.x和EntityFramework Core插入数据探讨的更多相关文章
- SQLAlchemy Core插入数据,有好几种方法呢
看是一次插入一条还是多条, 看是数据表名是变量还是常量, 操作还是很灵活的, 主要看哪种顺手顺眼啦. #coding=utf-8 from datetime import datetime from ...
- [MSSQL] [EntityFramework(.Net Core)] 自增长id字段,无法插入数据
IDENTITY_INSERT 为 OFF,无法插入数据, 类似的错误,解决记录: 网上查了下,都是 Code First 模式下的解决方案, 如:在 DBContext 的 OnModelCreat ...
- Webservice WCF WebApi 前端数据可视化 前端数据可视化 C# asp.net PhoneGap html5 C# Where 网站分布式开发简介 EntityFramework Core依赖注入上下文方式不同造成内存泄漏了解一下? SQL Server之深入理解STUFF 你必须知道的EntityFramework 6.x和EntityFramework Cor
Webservice WCF WebApi 注明:改编加组合 在.net平台下,有大量的技术让你创建一个HTTP服务,像Web Service,WCF,现在又出了Web API.在.net平台下, ...
- IdentityServer4 中文文档 -16- (快速入门)使用 EntityFramework Core 存储配置数据
IdentityServer4 中文文档 -16- (快速入门)使用 EntityFramework Core 存储配置数据 原文:http://docs.identityserver.io/en/r ...
- EF批量插入数据(Z.EntityFramework.Extensions)
EF用原生的插入数据方法DbSet.ADD()和 DbSet.AddRange()都很慢.所以要做大型的批量插入只能另选它法. 1.Nugget 2.代码 using EF6._0Test.EF; u ...
- 你必须知道的EntityFramework 6.x和EntityFramework Core变更追踪状态
前言 只要有时间就会时不时去看最新EF Core的进展情况,同时也会去看下基础,把握好基础至关重要,本节我们对比看看如标题EF 6.x和EF Core的不同,希望对正在学习EF Core的同行能有所帮 ...
- EntityFramework 6.x和EntityFramework Core必须需要MultipleActiveResultSets?
前言 本节我们来探讨到底需不需要在连接字符串上加上MultipleActiveResultSets = true ?,若您有更深层次的理解欢迎留下您的脚印. EntityFramework 6.x和E ...
- 你所不知道的库存超限做法 服务器一般达到多少qps比较好[转] JAVA格物致知基础篇:你所不知道的返回码 深入了解EntityFramework Core 2.1延迟加载(Lazy Loading) EntityFramework 6.x和EntityFramework Core关系映射中导航属性必须是public? 藏在正则表达式里的陷阱 两道面试题,带你解析Java类加载机制
你所不知道的库存超限做法 在互联网企业中,限购的做法,多种多样,有的别出心裁,有的因循守旧,但是种种做法皆想达到的目的,无外乎几种,商品卖的完,系统抗的住,库存不超限.虽然短短数语,却有着说不完,道不 ...
- 我的EntityFramework(2):简单的数据查询
原文:我的EntityFramework(2):简单的数据查询 在上一篇博文中,已经搭建了基本的框架,接下来就进行简单的数据查询,这里主要用了Linq 常见的数据集查询 var companyList ...
随机推荐
- javascript之BOM地址栏对象(Location)
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- (四十四)TabBarController和NagivationController配合
如果既要使用TabBar分页,又要使用Nagivation导航,那么只能是TabBar为根,Nagivation为TabBar子视图,每一个TabBar对应一个Nagivation导航的一系列页面. ...
- hadoop学习要点
一.HDFS (一)HDFS 概念 (二)HDFS命令行接口 (三)Java 接口 (四)文件读取和文件写入,一致性 (五)集群数据的均衡 (六)存档 (七)NameNode 单点故障问题 (八)大量 ...
- MPEG-7 视觉描述符
本文节选自<基于MPEG-7与内容的图像检索技术的研究>. MPEG-7 标准中视觉描述工具包括基本结构和描述符.本文主要介绍各描述符. (1)颜色描述符 MPEG-7 主要定义了七种颜色 ...
- SVN版本控制器中各符号的含义
SVN符号的含义 项目开发过程中,随着学习的不断深入,开始慢慢接触到版本管理控制工具,其实这个工具主要用于团队开发之中,但对于个人项目的备份也有好处,可以避免在电脑出现不可预知的故障时,最大化的保护自 ...
- Android ROM开发(三)——精简官方ROM并且内置ROOT权限,开启Romer之路
Android ROM开发(三)--精简官方ROM并且内置ROOT权限,开启Romer之路 相信ROM的相关信息大家通过前几篇的学习都是有所了解了,这里就不在一一提示了,这里我们下载一个官方包,我们还 ...
- Oracle数据库容灾备份技术探讨
Oracle数据库容灾备份技术探讨 三种Oracle灾备技术 对于Oracle数据库的灾备技术,我们可以从Data Guard,GoldenGate和CDP角度去考虑. Oracle Data Gua ...
- 安卓学习笔记--已root设备应用请求root权限
网上查到的方法 Process process = null; DataOutputStream os = null; try { String cmd = "chmod 777 " ...
- VB 对象未注册
像往常一样,打开我的VB,开始我的学生信息管理系统,突然弹出如下对话框: ...
- SharePoint 调用WebService操作List小记
简述:在SharePoint的使用中,经常需要进行系统集成这样的操作,我们作为SharePoint开发,就需要给其他系统提供接口,而SharePoint提供的WebService就很好的提供了这样的功 ...