Linq中小心使用IndexOf

 

我们平常在做字符串的模糊查询时,有可能会用到下面的类似LINQ写法:

string.IsNullOrEmpty(_SN) ? true : a.SN.IndexOf(_SN) != -1

 

这条LINQ翻译为SQL如下:

( ( CASE

                 WHEN ( ( @p__linq__1 IS NULL )

                         OR ( ( Cast(Len(@p__linq__1) AS INT) ) = 0 ) ) THEN Cast(1 AS BIT)

                 WHEN ( NOT ( ( -1 = ( ( Cast(Charindex(@p__linq__2, [Extent1].[SN]) AS INT) ) - 1 ) )

                              AND ( ( Cast(Charindex(@p__linq__2, [Extent1].[SN]) AS INT) ) - 1 IS NOT NULL ) ) ) THEN Cast(1 AS BIT)

                 WHEN ( -1 = ( ( Cast(Charindex(@p__linq__2, [Extent1].[SN]) AS INT) ) - 1 ) ) THEN Cast(0 AS BIT)

               END ) = 1 )

 

注意看黄色部分,这条翻译过来的语句出了问题。

 

附:CHARINDEX用法

expression2 中搜索 expression1 并返回其起始位置(如果找到)。搜索的起始位置为 start_location

如果 expression1 expression2 之一为 NULL,并且数据库兼容级别为 70 或更高,则 CHARINDEX 将返回 NULL。如果数据库兼容级别为 65 或更低,则 CHARINDEX 将仅在 expression1 expression2 都为 NULL 时才返回 NULL 值。

如果在 expression2 内找不到 expression1,则 CHARINDEX 返回 0

返回的开始位置从 1 开始,而非从 0 开始。

分析以下几种情况:

 

表达式2为空白值的情况

SELECT ( CASE

           WHEN ( ( '123456' IS NULL )

                   OR ( ( Cast(Len('123456') AS INT) ) = 0 ) ) THEN Cast(1 AS BIT)

           WHEN ( NOT ( ( -1 = ( ( Cast(Charindex('123456', '') AS INT) ) - 1 ) )

                        AND ( ( Cast(Charindex('123456', '') AS INT) ) - 1 IS NOT NULL ) ) ) THEN Cast(1 AS BIT)

           WHEN ( -1 = ( ( Cast(Charindex('123456', '') AS INT) ) - 1 ) ) THEN Cast(0 AS BIT)

         END )

 

结果为0

 

表达式2为空值(NULL)的情况

 

SELECT ( CASE

           WHEN ( ( '123456' IS NULL )

                   OR ( ( Cast(Len('123456') AS INT) ) = 0 ) ) THEN Cast(1 AS BIT)

           WHEN ( NOT ( ( -1 = ( ( Cast(Charindex('123456', null) AS INT) ) - 1 ) )

                        AND ( ( Cast(Charindex('123456', null) AS INT) ) - 1 IS NOT NULL ) ) ) THEN Cast(1 AS BIT)

           WHEN ( -1 = ( ( Cast(Charindex('123456', null) AS INT) ) - 1 ) ) THEN Cast(0 AS BIT)

         END )

 

结果为1

 

测试其关键部分

SELECT ( CASE

           WHEN ( NOT(( -1 = ( ( Cast(Charindex('123456', null) AS INT) ) - 1 ) )

                        AND ( ( Cast(Charindex('123456', null) AS INT) ) - 1 IS NOT NULL ) ) ) THEN Cast(1 AS BIT)

         END )

结果为1

 

也就是说,当表中要查找的列中存在空值时,用indexOf会把所有空值列查出来。

 

其原因就是上文所说的黄色部分,注意那个NOT,它放错了位置!

正确方式如下:

SELECT ( CASE

           WHEN ( ( '123456' IS NULL )

                   OR ( ( Cast(Len('123456') AS INT) ) = 0 ) ) THEN Cast(1 AS BIT)

           WHEN ( ( NOT( -1 = ( ( Cast(Charindex('123456', null) AS INT) ) - 1 ) )

                        AND ( ( Cast(Charindex('123456', null) AS INT) ) - 1 IS NOT NULL ) ) ) THEN Cast(1 AS BIT)

           WHEN ( -1 = ( ( Cast(Charindex('123456', null) AS INT) ) - 1 ) ) THEN Cast(0 AS BIT)

         END )

 

结果为NULL

 

结论:

当我们在模糊查询时,如果确定要查的列没有NULL值,可以使用indexOf,否则要考虑其它方式,比如Contains(它翻译的SQLLIKE)。

转载于:https://www.cnblogs.com/wdf2gy/p/4366872.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值