EFCore Owned Entity Types,彩蛋乎?鸡肋乎?之鸡肋篇
鸡肋###
鸡肋(Chicken ribs),现代汉语词语,出自《三国志·魏书·武帝纪》裴松之注引《九州春秋》曰:“夫鸡肋,弃之如可惜,食之无所得,以比汉中,知王欲还也。”(食之无肉,弃之不舍) https://baike.baidu.com/item/鸡肋/206189
相信小伙伴们都知道这个词的典故。
那么,继续上一个彩蛋篇,让我来继续解释一下, 鸡肋乎 的说法,又是怎么一回事。
(本篇例子的源代码,可以在 https://github.com/kentliu2007/EFCoreDemo/tree/master/OwnedEntityTypes 下载。建议可以下载之后对照着代码来阅读本篇。我用的是VS2017)
为什么我们要做Table Split####
为什么我们要做Table Split?一切都是为了性能。
当今的关系型数据库产品,例如Oracle,或者SQL Server,哪怕你的Select命令,只返回一条记录里面的一个字段的内容,它的数据库引擎,都会从磁盘读取最小单位的块(Block)到内存里,然后再从内存里的N条记录中,过滤出你要的那条记录的那个字段,然后返回给你。
所以如果我们的数据表,每行的长度很长的情况下(例如拥有很多很多的字段,或者某几个字段特别的长),一个最小单位的块,能保存的记录行数就相对的少了。然后如果我们要查询和计算的记录行数比较多的话,数据库需要从磁盘读取的块,就相应地多了很多。从而影响性能。
针对这种情况,我们通常会对数据表进行纵向切割。对表的字段,按照某个方式,分门别类地,陆陆续续地把某些字段分离到另外一些表里面。例如,我们可以把存放着地址的那些长字段弄到另外一个表,当程序需要用到地址资料的时候才去访问这个表。这样,性能就提升上来了。
这个做法,就是Table Split。
EFCore Owned Entity Types对Table Split的支持####
如果看了上一篇,大家就可以知道,通过两个Entity,分别ToTable到两个不同的Table,且最后我们是统一通过Client这个类来访问所有属性,其实已经实现了对Table Split的支持。
当然,通过访问Client.ContactInfo属性才可以访问到电话号码、电子邮件,这样的方式有点不够便利。所以我们还可以进一步通过捣弄 getter/setter的把戏,直接通过Client.CellPhoneNo或者Client.Email的方式来获取联系资料。
具体请看下面的程序:
- 项目

- 代码




看,这样处理之后,是不是很方便呢?
性能####
等等,刚才不是说了,做Table Split就是为了性能吗?主要是我们希望可以纵向切割数据表之后,程序在需要的时候,才去访问相应的数据表,通过这个做法来提升性能。例如如果程序没有需要访问地址资料的需求,那么就不会去访问存放地址资料的数据表。
Owned Entity Types有能够做到这个效果吗?让我们用SQL Server Profiler来监视一下,上面的程序,它是怎样访问数据表的。
上面的Unit Test程序,实际上,它发出是SQL命令如下:

- 前两条Insert命令,没毛病:


- 最后两条Delete,也没毛病


- 中间的那条Select命令,就透露着 诡异 了:

这个相当于在程序里面用了Include()命令的做法,是Eager Loading啊。
那么没有实现程序在需要的时候,才去访问相应数据表的效果啊。
其实我们是希望在程序运行到
var x = context.Clients.FirstOrDefault(e => e.Code == code);
的时候,它只访问Client表。
然后接着程序运行到
Assert.AreEqual(x != null && x.Email == email, true);
的时候,它才去访问ClientContactInfo表。
现在的情况是,当运行context.Clients.FirstOrDefault(e => e.Code == code);的时候,其实它就已经用Left Join的方式,把Client表和ClientContactInfo两个表都同时访问了。
所以如果用了Owned Entity Types来支持Table Split的话,对性能是没有提升的。
在我之前的EFCore系列的博客里,有一篇如何用EFCore Lazy Loading实现Entity Split。通过阅读那篇博客,你就可以看到,用EFCore的Lazy Loading,其实我们是可以用着同样的捣弄getter/setter的伎俩,通过walk around的形式来支持Table Split,同时实现了程序在需要的时候,才去访问相应数据表,从而达到原先希望提升系统性能的初衷。(当然,也建议你去下载之前的代码,自己去运行SQL Server Profiler来验证一下EFCore Lazy Loading的运行机制)
所以,对比着我的上一篇博客,我们回过头来看Owned Entity Types,同样是支持Table Split,但是用Lazy Loading的话有着更好的系统性能,那么,你有没有感觉到 鸡肋 的感觉呢?
EFCore Owned Entity Types,彩蛋乎?鸡肋乎?之鸡肋篇的更多相关文章
- EFCore Owned Entity Types,彩蛋乎?鸡肋乎?之彩蛋篇
EFCore Owned Entity Types的定义 EFCore Owned Entity Types的文档在这里:https://docs.microsoft.com/zh-cn/ef/cor ...
- EF Core 新特性——Owned Entity Types
Owned Entity Types 首先owned entity type是EF Core 2.0的新特性. 至于什么是owned entity types,可以先把他理解为EF Core官方支持的 ...
- 实现了一个百度首页的彩蛋——CSS3 Animation简介
在百度搜索中有这样一个彩蛋:搜索“旋转”,“跳跃”,“反转”等词语,会出现相应的动画效果(搜索“反转”后的效果).查看源码可以发现,这些效果正是通过CSS3的animation属性实现的. 实现这个彩 ...
- Spring Boot -- 启动彩蛋
使用Spring Boot启动的jar包总是会显示一个Spring的图标: . ____ _ __ _ _ /\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \ ( ( )\_ ...
- Chrome 中的彩蛋,一款小游戏,你知道吗?
今天看到一篇文章,介绍chrome中的彩蛋,带着好奇心进去看了一眼,没想到发现了一款小游戏,个人觉得还不错,偶尔可以玩一下,放松放松心情!^_^ 当 Chrome 无法连接到互联网时, 或者上着网突然 ...
- 淘宝首页源码藏美女彩蛋(上)(UED新作2013egg)
今日,偶尔翻看淘宝源码,发现竟有美女形状源码.如下图: 此段代码在console中运行,结果更为惊叹. 亲手尝试的读者已经看到了代码运行的结果.taobao.com的console打印出了UED的招聘 ...
- 淘宝首页源码藏美女彩蛋(下)(UED新作2013egg)
我们已经知道,执行美女会得到"彩蛋",而正是彩蛋做到了taobaoUED展现给大家的神奇的前端魅力.今天我们来看看FP.egg&&FP.egg("%cjo ...
- [深入浅出Windows 10]模拟实现微信的彩蛋动画
9.7 模拟实现微信的彩蛋动画 大家在玩微信的时候有没有发现节日的时候发一些节日问候语句如“情人节快乐”,这时候会出现很多爱心形状从屏幕上面飘落下来,我们这小节就是要模拟实现这样的一种动画效果.可能微 ...
- 阻止PHP彩蛋信息泄漏 [转]
Easter Eggs(复活节彩蛋)外行人估计不了解这是神木玩意,彩蛋的网络解释是:用于电脑.电子游戏.电脑游戏.影碟或其他互动多媒体之中的隐藏功能或信息.PHP包含一个安全漏洞,可能导致未经授权的信 ...
随机推荐
- springboot实现自定义的错误页面展示
https://blog.csdn.net/trusause/article/details/84299886 参考 SpringBoot默认的错误处理机制 默认效果为: 返回一个默认的错误页面 Wh ...
- JavaWeb网上商城项目中用户注册,使用MailServer和FoxMail搭建本地邮件服务器
下载并安装易邮邮件服务器MailServer和腾讯邮箱FoxMail,下载地址 https://download.csdn.net/download/checkerror2/10130538 具体步 ...
- luogu准备复习(学习)题单
矩阵乘法 P1306 exbsgs P4195 网络流(割点) P1345 主席树 P3302
- IPTABLES使用总结(内网模拟银行网络)
iptables中有以下三种类型的表: FILTER表,默认的表,包含以下三种内建链: INPUT链,发给本地sockets的包 FORWARD链,经由系统发送的包 OUTPUT链,本地生成并发出的包 ...
- vue中import引入模块路径中@符号是什么意思
在编写vue文件中引入模块 import model from "@/common/model"; 这里路径前面的“@”符号表示什么意思? resolve: { // 自动补全的扩 ...
- mysql查表的时候报错:java.sql.SQLException: Value '0000-00-00 00:00:00' can not be represented as java.sql.Timest
在spark项目中读取mysql中的数据的时候,发生了报错: 原因:主要是因为数据库中有一些字段为null,无法进行转换: 上面字段那种,我的inserttime字段类型为timestamp类型,不为 ...
- 使用 Laragon 在 Windows 中快速搭建 Laravel 本地开发环境 (转)
laravel学院 简介 对于那些使用 Windows 操作系统的同学来说,Homestead 和 LaraDock 虽说支持 Windows 系统,但是对初学者来说,安装配置起来还是有一定复杂度的, ...
- 20165309 《网络对抗技术》实验五:MSF基础应用
20165309 <网络对抗技术>实验五:MSF基础应用 1.基础问题回答 (1)什么是exploit? (2)什么是payload? (3)什么是encode? (4)离实战还缺些什么技 ...
- live 2d js demo
<!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...
- java知识随笔
Servlet: void init(ServletConfig var1) throws ServletException; ServletConfig getServletConfig(); vo ...