热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

开发笔记:讨论过后而引发对EF6.x和EFCore查询缓存的思考

本文由编程笔记#小编为大家整理,主要介绍了讨论过后而引发对EF6.x和EFCore查询缓存的思考相关的知识,希望对你有一定的参考价值。前言
本文由编程笔记#小编为大家整理,主要介绍了讨论过后而引发对EF 6.x和EF Core查询缓存的思考相关的知识,希望对你有一定的参考价值。



前言

最近将RabbitMQ正式封装引入到.NET Core 2.0项目当中,之前从未接触过这个高大上的东东跟着老大学习中,其中收获不少,本打算再看看RabbitMQ有时间写写,回来后和何镇汐大哥探讨了一点关于EF和EF Core的内容,于是乎本文就出来了。EF 6.x和EF Core中的查询缓存想必大家都有耳闻或者了解,从数据库中查询出来的实体会形成快照在内存中Copy一份且被上下文跟踪,接下来我们要讲的内容就是这个,我们来看看。


EF 6.x和EF Core查询缓存思考

首先我利用EF Core通过一个例子来进入今天的主题,在此过程中您也可多一点思考空间并对照和您想象中的结果是否有出入或者不一致的地方。



技术分享图片

var cOntext= new EFCoreDbContext();
var blog1 = context.Blogs.Find(3);
Console.WriteLine($"数据库原始值为{blog1.Name}");
blog1.Name = "Jeffcky";
var blog2 = context.Blogs.FirstOrDefault(d => d.Id == 3);
Console.WriteLine($"查询数据库后的值为{blog2.Name}"

技术分享图片

上述我们通过Find方法查询出Blog1,然后对Name进行赋值,紧接着我们再次查询主键等于3的实体,您猜想一下blog2中的Name会等于多少呢?

技术分享图片

我们看看如上图所示结果为Jeffcky,有的童鞋就问了,我们进行第一次查询时值为3,然后我们仅仅只是赋值为Jeffcky并未提交,当我们再次查询相同实体时结果却为Jeffcky呢?难道不应该是3么,对吧。稍微对EF或者EF Core有所了解的大佬们明白第一次查询时会有快照,当下次查询时EF或者EF Core会在内存中根据第一次查询所对应的哈希值和主键去查找,此时查找到则直接利用内存中的对象,所以此时blog2的Name为Jeffcky。问题是不是到此就结束了呢?如果是这样,那我大半夜还浪费这时间写这篇博客!接下来我们再来看看两次查询所生成的SQL如何?

技术分享图片

技术分享图片

在EF 6.x和EF Core中 通过Find方法基于主键查询查询可重用,什么意思呢?如上第一个查询采用参数化查询,也就说如果我们下次再利用Find方法查询那么将不会到数据库查询,而是直接从内存中返回,有的童鞋就想了,恩,挺好,这样显著提高了查询性能,我只能说不一定看对应场景,如果对于非常频繁的查询我个人觉得不建议用此方法,因为还有对实体的更新操作啊,此时数据更新了却在内存中的值没有更新显示到UI上也就是说是过期了值,那么您觉得这个时候用Find方法还可取吗。对于第二个查询则直接采取赋值的形式(在我即将出版的《你必须掌握的EntityFramework 6.x和Core 2.0》书中有讲解EF 6.x查询的很大问题)这都不是事,问题是第二次我们利用FirstOrDefault方法查询此时居然走数据库了,不信,您可以看看如下利用SQL Profiler监控得到的SQL语句。

技术分享图片

这就让人有点费解了,第二个查询既然是到数据库中查询那为何我们得到的值当前第一次查询出来修改但未被提交的值呢?这不是自相矛盾么,EF Core这样设计的意义何在,我也想不通,我能想到的是在同一上下文中大部分情况下不会对同一实体查询多次,但是谁能保证呢。 在EF Core中除了Find会进行翻译缓存,其他比如First、FirstOrDefault、Last、LastOrDefault都会到数据库中查询,当我们利用Last或者LastOrDefault查询时会出现更有意思的事情,我们看看生成的SQL语句:


var blog3 = context.Blogs.Last();

技术分享图片

EF Core大哥哥们这么简单的查询问题都没测试到么,你翻译给我让我误以为这是返回所有列表了,结果一看查询出来的值却是对的,我想这是EF Core大哥哥们忘记加上TOP 1和ORDER BY了,到github上一搜索原来有大神提过这个ISSUE(https://github.com/aspnet/EntityFrameworkCore/issues/10493)在2.1版本发布会解决这个问题。好了我们回到主题所遇到的问题,在同一上下文查询同一实体,第一次查询如果我们利用Find方法查询会进行翻译缓存,待下一次再次查询时不会到数据库中查询,如果是下一次查询不是利用Find方法查询,比如FirstOrDefault此时会到数据库中查询但是此时的值却不是数据库中的值而是当前被修改而未被提交的值,那么我们如何获取数据库中的值而不是当前修改而未被提交的值呢?请往下看。


EF和EF Core获取数据库值而不是当前修改未被提交的值

我们讨论了问题的出现,接下来我们尝试利用方法来解决,我们可以对上下文所跟踪的实体进行移除通过Local.Remove,如下:



技术分享图片

var cOntext= new EFCoreDbContext();
var blog1 = context.Blogs.Find(3);
Console.WriteLine($"数据库原始值为{blog1.Name}");
blog1.Name = "Jeffcky";
context.Blogs.Local.Remove(blog1);
var blog2 = context.Blogs.FirstOrDefault(d => d.Id == 3);
Console.WriteLine($"查询数据库后的值为{blog2.Name}");

技术分享图片

Local方法意味获取在本地所添加、修改等的实体,我们获取被上下文所本地被上下文所跟踪的实体然后移除,继而再进行查询是不是就可以移除呢?不好意思移除不了。

技术分享图片


利用AsNoTracking方法

这个算是最简单的方法之一了,仅仅对于查询而言通过此方法不会形成快照从而提高查询性能。



技术分享图片

var cOntext= new EFCoreDbContext();
var blog1 = context.Blogs.Find(3);
Console.WriteLine($"数据库原始值为{blog1.Name}");
blog1.Name = "Jeffcky";
var blog2 = context.Blogs.AsNoTracking().FirstOrDefault(d => d.Id == 3);
Console.WriteLine($"查询数据库后的值为{blog2.Name}");

技术分享图片

技术分享图片


将实体状态标记为Detached不被上下文跟踪



技术分享图片

var cOntext= new EFCoreDbContext();
var blog1 = context.Blogs.Find(3);
Console.WriteLine($"数据库原始值为{blog1.Name}");
blog1.Name = "Jeffcky";
context.Entry(blog1).State = EntityState.Detached;
var blog2 = context.Blogs.FirstOrDefault(d => d.Id == 3);
Console.WriteLine($"查询数据库后的值为{blog2.Name}");

技术分享图片


 通过Reload方法刷新实体



技术分享图片

var cOntext= new EFCoreDbContext();
var blog1 = context.Blogs.Find(3);
Console.WriteLine($"数据库原始值为{blog1.Name}");
blog1.Name = "Jeffcky";
context.Entry(blog1).Reload();
var blog2 = context.Blogs.FirstOrDefault(d => d.Id == 3);
Console.WriteLine($"查询数据库后的值为{blog2.Name}");

技术分享图片


通过GetDatabaseValues方法直接获取数据库中值



技术分享图片

var cOntext= new EFCoreDbContext();
var blog1 = context.Blogs.Find(3);
Console.WriteLine($"数据库原始值为{blog1.Name}");
blog1.Name = "Jeffcky";
var blog2 = (Blog)context.Entry(blog1).GetDatabaseValues().ToObject();
Console.WriteLine(blog2.Name);
//或者
var db = context.Entry(blog1).GetDatabaseValues();
Console.WriteLine(db["Name"]);

技术分享图片


总结

好了今天的内容就到此为止了,无论是EF 6.x还是EF Core,只要我们对一些原理足够了解才不至于出现让人意想不到的问题。希望本文对您有所帮助,下节开始讲讲RabbitMQ,我们下节再会。




推荐阅读
  • 本文深入探讨了 MXOTDLL.dll 在 C# 环境中的应用与优化策略。针对近期公司从某生物技术供应商采购的指纹识别设备,该设备提供的 DLL 文件是用 C 语言编写的。为了更好地集成到现有的 C# 系统中,我们对原生的 C 语言 DLL 进行了封装,并利用 C# 的互操作性功能实现了高效调用。此外,文章还详细分析了在实际应用中可能遇到的性能瓶颈,并提出了一系列优化措施,以确保系统的稳定性和高效运行。 ... [详细]
  • 深入解析 Django 中用户模型的自定义方法与技巧 ... [详细]
  • 本文详细探讨了Java集合框架的使用方法及其性能特点。首先,通过关系图展示了集合接口之间的层次结构,如`Collection`接口作为对象集合的基础,其下分为`List`、`Set`和`Queue`等子接口。其中,`List`接口支持按插入顺序保存元素且允许重复,而`Set`接口则确保元素唯一性。此外,文章还深入分析了不同集合类在实际应用中的性能表现,为开发者选择合适的集合类型提供了参考依据。 ... [详细]
  • 利用C#技术实现Word文档的动态生成与编辑
    本文通过一个简单的示例,介绍了如何使用C#语言实现Word文档的动态生成与编辑功能。文章详细阐述了在项目中引用Word动态库的方法,并通过具体代码示例展示了如何创建和操作Word表格。此内容旨在为初学者提供参考和学习资料,欢迎读者提出宝贵意见和建议。 ... [详细]
  • 结语 | 《探索二进制世界:软件安全与逆向分析》读书笔记:深入理解二进制代码的逆向工程方法
    结语 | 《探索二进制世界:软件安全与逆向分析》读书笔记:深入理解二进制代码的逆向工程方法 ... [详细]
  • 成功实现Asp.Net MVC3网站与MongoDB数据库的高效集成
    我们成功地构建了一个基于Asp.NET MVC3框架的网站,并实现了与MongoDB数据库的高效集成。此次更新不仅完善了基本的创建和显示功能,还全面实现了数据的增删改查操作。在创建功能方面,我们修复了之前代码中的错误,确保每个属性都能正确生成。此外,我们还对数据模型进行了优化,以提高系统的性能和稳定性。 ... [详细]
  • 本文深入探讨了 iOS 开发中 `int`、`NSInteger`、`NSUInteger` 和 `NSNumber` 的应用与区别。首先,我们将详细介绍 `NSNumber` 类型,该类用于封装基本数据类型,如整数、浮点数等,使其能够在 Objective-C 的集合类中使用。通过分析这些类型的特性和应用场景,帮助开发者更好地理解和选择合适的数据类型,提高代码的健壮性和可维护性。苹果官方文档提供了更多详细信息,可供进一步参考。 ... [详细]
  • 在开发系统查询搜索功能时,需注意以下几点以提高信息检索效率:首先,在SQL语句中,每个参数占位符“?”后必须紧跟相应的参数赋值,确保参数与赋值一一对应,避免因参数不匹配导致的错误。其次,进行模糊搜索时,若用户输入通配符“%”,可能会导致全表扫描,因此需要对输入的“%”进行特殊处理或限制,以防止不必要的性能开销。此外,建议使用索引优化查询速度,并合理设计搜索逻辑,以提升用户体验。 ... [详细]
  • Spring Boot 实战(一):基础的CRUD操作详解
    在《Spring Boot 实战(一)》中,详细介绍了基础的CRUD操作,涵盖创建、读取、更新和删除等核心功能,适合初学者快速掌握Spring Boot框架的应用开发技巧。 ... [详细]
  • 在《PHP应用性能优化实战指南:从理论到实践的全面解析》一文中,作者分享了一次实际的PHP应用优化经验。文章回顾了先前进行的一次优化项目,指出即使系统运行时间较长后出现的各种问题和性能瓶颈,通过采用一些通用的优化策略仍然能够有效解决。文中不仅详细阐述了优化的具体步骤和方法,还结合实例分析了优化前后的性能对比,为读者提供了宝贵的参考和借鉴。 ... [详细]
  • 本文首先对信息漏洞的基础知识进行了概述,重点介绍了几种常见的信息泄露途径。具体包括目录遍历、PHPINFO信息泄露以及备份文件的不当下载。其中,备份文件下载涉及网站源代码、`.bak`文件、Vim缓存文件和`DS_Store`文件等。目录遍历漏洞的详细分析为后续深入研究奠定了基础。 ... [详细]
  • 本项目在Java Maven框架下,利用POI库实现了Excel数据的高效导入与导出功能。通过优化数据处理流程,提升了数据操作的性能和稳定性。项目已发布至GitHub,当前最新版本为0.0.5。该项目不仅适用于小型应用,也可扩展用于大型企业级系统,提供了灵活的数据管理解决方案。GitHub地址:https://github.com/83945105/holygrail,Maven坐标:`com.github.83945105:holygrail:0.0.5`。 ... [详细]
  • 本文介绍了使用 Python 编程语言高效抓取微博文本和动态网页图像数据的方法。通过详细的示例代码,展示了如何利用爬虫技术获取微博内容和动态图片,为数据采集和分析提供了实用的技术支持。对于对网络数据抓取感兴趣的读者,本文具有较高的参考价值。 ... [详细]
  • 在 Android 开发中,通过合理利用系统通知服务,可以显著提升应用的用户交互体验。针对 Android 8.0 及以上版本,开发者需首先创建并注册通知渠道。本文将详细介绍如何在应用中实现这一功能,包括初始化通知管理器、创建通知渠道以及发送通知的具体步骤,帮助开发者更好地理解和应用这些技术细节。 ... [详细]
  • 本课程详细介绍了如何使用Python Flask框架从零开始构建鱼书应用,涵盖高级编程技巧和实战项目。通过视频教学,学员将学习到Flask的高效用法,包括数据库事务处理和书籍交易模型的实现。特别感谢AI资源网提供的课程下载支持。 ... [详细]
author-avatar
____L振豪
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有