Entity Framework Core 大小写敏感处理

 

可以使用'StringComparison'吗?

在数据库查询操作中,不可避免去考虑字母大小写的问题,比如要在Movie表中查找“X-Men”这部电影,为了不区分字母大小写,按照Linq to memory的习惯,可能会写出如下代码:

DbContext.DbSet<Movie>
  .Where(item => string.Equals(item.Title, "X-Men", StringComparison.InvariantCultureIgnoreCase)

但是上述代码执行会报错,提示 'StringComparison'参数不支持。

InvalidOperationException: The LINQ expression 'DbSet<Movie>() .Where(m => string.Equals( a: m.Genre, b: __MovieGenre_0, comparisonType: InvariantCultureIgnoreCase))' could not be translated. 
Additional information: Translation of the 'string.Equals' overload with a 'StringComparison' parameter is not supported. 
See https://go.microsoft.com/fwlink/?linkid=2129535 for more information. 
Either rewrite the query in a form that can be translated, or switch to client evaluation explicitly by inserting a call to 'AsEnumerable', 'AsAsyncEnumerable', 'ToList', or 'ToListAsync'. 
See https://go.microsoft.com/fwlink/?linkid=2101038 for more information.

怎么解决这个问题呢?如果你用的是SQL server或MySQL,那么很简单,去掉“StringComparison.InvariantCultureIgnoreCase”这个参数就可以了,因为他们是默认大小写不敏感的。如果用的是大小写敏感的数据库,比如PostgreSQL,就需要做一些特殊处理了,后面会讲到。

对于这类问题有一个暴力的解决方法,就是按报错提示的那样在查询客户端做筛选:

DbContext.DbSet<Movie>
  .ToList()
  .Where(item => string.Equals(item.Title, "X-Men", StringComparison.InvariantCultureIgnoreCase)

这样做的本质是返回所有的Movie,然后在内存中做Title匹配查询(Linq to memory),而不是在DB中做Title的匹配查询,所以不会报 'StringComparison'参数不支持的错误。这种方法在绝大多数场景下是不推荐的,因为 1. 性能差,查询是在内存中不是在DB中,2. 占用内存,返回了整张表

 

为什么不能使用'StringComparison'?

前面讲到有的数据库是默认大小写不敏感的:SQL server,MySQL;有的数据库是默认大小写敏感的:PostgreSQL。那么数据库中大小写敏感与否是由什么控制的呢,排序规则(Collation,一组规则,用于确定文本值的排序和相等性比较方式,可以在数据库,表和列上创建排序规则,排序规则是隐式继承的,比如在数据库上创建了排序规则,没有在表上创建排序规则,那么表会默认使用数据库的排序规则,列同理)。因为EF core不知道数据库,表,列支持/应用了什么样的排序规则,所以'StringComparison'是没有意义的,EF core会将  string.Equals 转化成数据库的相等( = )操作,由数据库根据列上应用的排序规则来决定是否区分大小写。

 

如何在查询中设置区分大小写与否?

既然区分大小写是由排序规则决定的,我们可以通过在查询中指定排序规则的方式来来设置是否区分大小写,例如SQL Server默认是不区分大小写的,我们可以通过 EF.Functions.Collate 来指定一个大小写敏感的排序规则"SQL_Latin1_General_CP1_CS_AS"来达到精确匹配的目的

DbContext.DbSet<Movie>
    .Where(m => EF.Functions.Collate(item.Title, "SQL_Latin1_General_CP1_CS_AS") == "X-Men")

但是这种显示指定排序规则的方法也不是非常推荐的,因为他会导致索引匹配失败,进而影响查询性能。索引隐式继承列上的排序规则,当显示指定的排序规则和创建索引时指定的排序规则(PostgreSQL支持在创建索引时指定排序规则)或列的排序规则不一致时,会导致索引匹配失败,进而导致查询不能应用索引,这也是EF Core没有将'StringComparison'转换成排序规则的另一个原因。

因此一个相对完善的解决方案是,根据业务模型在列上指定合适的排序规则,而不是在代码中设置。如果数据库支持在列上创建多个索引,你也可以用显示指定排序规则的方式根据业务场景切换排序规则来匹配正确的索引。

 

参考文档:

https://learn.microsoft.com/en-us/ef/core/miscellaneous/collations-and-case-sensitivity

https://learn.microsoft.com/en-us/sql/relational-databases/collations/collation-and-unicode-support?view=sql-server-ver16

https://www.postgresql.org/docs/current/collation.html

原文链接:https://www.cnblogs.com/Tiger-Lu/p/17311560.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:Entity Framework Core 大小写敏感处理 - Python技术站

(0)
上一篇 2023年4月19日
下一篇 2023年4月19日

相关文章

  • Unity登录注册时限制发送验证码次数功能的解决方法

    Unity登录注册时限制发送验证码次数功能的解决方法可以通过以下几步实现: 1. 设计数据库结构 在设计数据库时,为用户表添加两个属性字段,一个是上次发送验证码时间,一个是发送验证码次数。例如: CREATE TABLE users ( id INT PRIMARY KEY AUTO_INCREMENT, email VARCHAR(255) NOT NUL…

    C# 2023年6月3日
    00
  • 详解C# 网络编程系列:实现类似QQ的即时通信程序

    详解C#网络编程系列:实现类似QQ的即时通信程序 简介 本文将详细讲解如何使用C#网络编程实现类似QQ的即时通信程序,该程序基于TCP协议,使用Socket实现客户端与服务端的通信,涵盖了详细的代码实现和功能介绍。 实现步骤 设计即时通信协议。 实现服务端程序,包括监听和处理客户端请求。 实现客户端程序,包括连接和向服务端发送请求。 实现消息发送和接收功能。…

    C# 2023年5月15日
    00
  • C#生成唯一值的方法汇总

    生成GUID GUID是一种可以用于生成全球唯一标识符的算法,具有足够的随机性和唯一性。在C#中可以通过Guid.NewGuid()方法生成GUID。 Guid guid = Guid.NewGuid(); string uniqueId = guid.ToString(); 通过时间戳生成唯一值 根据当前时间计算其与一个固定日期之间的时间间隔的毫秒数,将其…

    C# 2023年6月1日
    00
  • DropDownList设置客户端事件思路

    下面是关于 DropDownList 设置客户端事件的完整攻略: 思路简介 DropDownList 是 ASP.NET WebForm 中常用的组件之一,在前端页面上展示一个下拉列表框,并且支持使用 C#、VB 等服务器端语言动态生成下拉列表内容。如果需要在前端页面使用 JS 代码对 DropDownList 进行操作,就需要用到客户端事件,这里主要指的是…

    C# 2023年5月31日
    00
  • C# 计算标准偏差相当于Excel中的STDEV函数实例

    计算标准偏差(Standard Deviation)是统计学中的一个重要概念,它是一组数据的离散程度的度量标准。在统计分析中,标准偏差用来衡量数据集中数据的分散情况。在C#中,可以使用Math库中的方法进行计算。下面是计算标准偏差相当于Excel中STDEV函数实例的完整攻略。 步骤一:定义数据 首先,我们需要定义一组数据作为计算标准偏差的数据源。假设我们有…

    C# 2023年6月8日
    00
  • C# .NET中Socket简单实用框架的使用教程

    C#.NET中Socket简单实用框架的使用教程 什么是Socket框架? Socket框架是一种网络编程模型,被广泛应用于各种网络通信应用程序的开发中,包括聊天工具、文件传输、视频会议等。Socket框架通过封装了底层的TCP/IP协议,提供了一种简单、灵活且高效的网络编程模型。通常情况下,使用Socket框架编写网络应用程序需要掌握C#语言和.NET框架…

    C# 2023年5月15日
    00
  • C++学习之Lambda表达式的用法详解

    C++学习之Lambda表达式的用法详解 在本文中,我们将详细介绍C++中Lambda表达式的用法和使用场景。 什么是Lambda表达式? Lambda表达式是一种在C++11标准中引入的新功能,它允许我们将一个函数作为一个变量来使用。Lambda表达式相当于一种无名函数,可以在需要的时候定义,并且和普通函数一样具有函数的特性,包括参数、返回值、局部变量等。…

    C# 2023年6月8日
    00
  • ASP.NET Core MVC通过IActionFilter.OnActionExecuting方法,获取Controller的Action方法参数值

    用过ASP.NET Core MVC中IActionFilter拦截器的开发人员,都知道这是一个非常强大的MVC拦截器。最近才发现IActionFilter的OnActionExecuting方法,甚至可以获取Controller的Action方法参数值。 假如我们在ASP.NET Core MVC项目中有一个HomeController,其中有一个Acti…

    C# 2023年4月24日
    00
合作推广
合作推广
分享本页
返回顶部