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日

相关文章

  • C#线程入门教程之单线程介绍

    下面我将详细讲解一下“C#线程入门教程之单线程介绍”的完整攻略。 1. 什么是线程? 在介绍单线程之前,我们先来了解一下什么是线程。在计算机中,线程是进程内部的一个独立执行流,用于执行并发任务。与进程不同,线程之间共享同一进程的内存和文件,可以方便地协同工作。 2. 单线程 单线程指的是程序只有一个线程在执行任务。在此种情况下,一个任务必须等待另一个任务完成…

    C# 2023年6月7日
    00
  • 详细分析ASP.NET Razor之C# 变量

    ASP.NET Razor是一种在.NET平台上构建Web应用程序的技术,其模板引擎支持C#编程,并提供了一些方便的快捷语法来简化代码书写。本文将详细讲解ASP.NET Razor之C#变量的使用方法以及示例说明。 1. 声明变量 在Razor模板中,可以使用@符号来声明一个C#变量,并将其输出到HTML。例如: @{ var name = "Al…

    C# 2023年6月3日
    00
  • C#调用usb摄像头的实现方法

    下面是详细的攻略: 准备工作 电脑需要安装相关的摄像头驱动程序。 了解 C# 操作 USB 设备的基础知识。 实现方法 导入 System.Management 和 System.Text.RegularExpressions 两个命名空间。 using System.Management; using System.Text.RegularExpressi…

    C# 2023年6月3日
    00
  • C#基础知识之字符串和正则表达式

    C#基础知识之字符串和正则表达式 一、字符串 1. 字符串的定义 在 C# 中,字符串是一个不可变的对象,表示文字和其他字符序列。C# 中的字符串对象是 System.String 类型的实例。创建字符串即是创建 String 对象,并使用双引号或 @-引号字符串来表示字符串值。如: string str1 = "Hello world!&quot…

    C# 2023年6月1日
    00
  • .Net Core2.1 WebAPI新增Swagger插件详解

    .Net Core2.1 WebAPI新增Swagger插件详解 Swagger是一种API文档工具,它可以自动生成API文档,并提供一个交互式的UI界面,方便开发人员测试API。在.Net Core2.1中,我们可以使用Swagger插件来自动生成API文档。本攻略将详细介绍如何使用Swagger插件。 安装Swagger插件 我们可以使用以下命令来安装S…

    C# 2023年5月17日
    00
  • C# Path.GetFileName – 获取路径中的文件名部分

    Path.GetFileName 方法是 .NET Framework 中提供的一个用于获取路径中文件名的静态方法,主要用于从路径中获取文件名,也可用于处理文件路径字符串,它位于 System.IO 命名空间中。 以下是该方法的作用: 获取给定路径字符串中的文件名。 通过计算要从中返回的位置的字符数,以及路径分隔符(注意:Windows 和 .NET 中的路…

    C# 2023年4月19日
    00
  • asp.net 文件下载的通用方法

    ASP.NET 是一个强大的 Web 应用程序框架,可用于创建各种类型的 Web 应用程序。如果你需要在应用程序中实现文件下载功能,下面是一套通用的方法,能够帮助你轻松达成目标。 使用 Response.TransmitFile 方法实现文件下载 第一种实现文件下载的方法是使用 Response.TransmitFile 方法。这种方法的优点是非常简单易用,…

    C# 2023年6月3日
    00
  • 解析C#中@符号的几种使用方法详解

    解析C#中@符号的几种使用方法详解 什么是@符号 在C#的语法中,@符号表示对一个字符串进行特殊处理。具体地讲,它可以将一个字符串中的转义字符忽略掉,也可以使一个字符串中的关键字变成非关键字。 @符号的使用方法 1. 忽略字符串中的转义字符 我们知道,在普通的字符串中,转义字符需要使用反斜杠进行转义,比如: string str1 = "这是一个带…

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