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/archive/2023/04/13/17311560.html

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

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

相关文章

  • C# 使用Dictionary复制克隆副本及比较是否相等

    下面我将详细讲解“C# 使用Dictionary复制克隆副本及比较是否相等”的完整攻略。 1. 使用Dictionary类型 首先,我们需要使用 C# 中的 Dictionary 类型来存储数据,这个类型可以看做是一种键值对的映射关系,其中的键和值均可以是任意类型。对于复制克隆副本和比较是否相等的操作,我们需要熟悉以下几个方法: 1.1. Add 方法 该方…

    C# 2023年5月31日
    00
  • 各种AJAX方法的使用比较详解

    AJAX(Asynchronous JavaScript and XML)是一种用于创建异步Web应用程序的技术。它可以在不刷新整个页面的情况下更新部分页面内容,提高Web应用程序的响应速度和用户体验。本文将介绍各种AJAX方法的使用,包括XMLHttpRequest、jQuery AJAX和Fetch API。 XMLHttpRequest XMLHttp…

    C# 2023年5月15日
    00
  • C#调用pyd的方法

    当我们需要使用Python库的时候,可以通过C#代码调用Python库提供的功能。Python库通常是以.so 或 .pyd 的文件形式提供,因此,我们需要使用C#的相关机制调用 Python库。下面将介绍如何在C#中调用Python库的方法。 步骤一: 安装Python 我们需要在计算机上安装Python,并添加Python的安装目录到系统路径中。可以通过…

    C# 2023年6月3日
    00
  • 浅谈c#表达式树Expression简单类型比较demo

    让我来详细讲解一下“浅谈c#表达式树Expression简单类型比较demo”的攻略。 什么是表达式树Expression? Expression是.NET Framework中定义的一个类,它代表了一个可执行的代码块。所谓的表达式树Expression就是将一段具体的代码逻辑抽象成树型结构, 如何使用表达式树实现简单类型比较? 表达式树可以用来构建动态查询…

    C# 2023年6月1日
    00
  • C#实现简单过滤非法字符实例

    下面是对该问题的详细讲解: 1.背景介绍 在许多场景下,需要对用户输入的数据进行过滤,以防止非法字符的出现。这时候一个比较常见的做法就是使用正则表达式对用户输入的字符串进行校验,屏蔽非法字符,这样既保证了数据的正确性,也提升了应用程序的安全性。 本篇攻略就是介绍如何使用C#编程语言实现简单的过滤非法字符功能。 2.实现过程 2.1 初步设计 在C#中,我们可…

    C# 2023年6月7日
    00
  • jquery中ajax调用json数据的使用说明

    在Web开发中,使用Ajax调用JSON数据是一种常见的任务,它可以帮助开发者动态地加载和更新页面内容。在本攻略中,我们将介绍如何使用jQuery中Ajax调用JSON数据,并提供两个示例来说明其用法。 以下是两个示例,介绍如何使用jQuery中Ajax调用JSON数据: 示例一:使用$.ajax方法调用JSON数据 首先,我们需要引入jQuery库: &l…

    C# 2023年5月15日
    00
  • Asp.Net Core配置多环境log4net配置文件的全过程

    在 ASP.NET Core 项目中,使用 log4net 记录日志是一种常见的方式。在多环境下,我们需要为每个环境配置不同的 log4net 配置文件。以下是 ASP.NET Core 配置多环境 log4net 配置文件的全过程: 步骤一:添加 log4net 包 首先,需要在 ASP.NET Core 项目中添加 log4net 包。可以使用 NuGe…

    C# 2023年5月17日
    00
  • C# 如何添加错误日志信息

    当我们在开发C#应用程序时,通常需要将错误日志信息输出到一个日志文件中,以便于在应用出现问题时能够及时定位错误并进行跟踪。本文将介绍如何在C#应用程序中添加错误日志信息。 1. 引入命名空间 using System.IO; 2. 创建日志文件 string logFilePath = @"C:\Logs\myLog.txt"; Stre…

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