如何解决WHERE 中的 SQL Server CASE 具有单个条件但多个返回 用户故事开发问题代码示例
用户故事
我需要选择列来检查空值,并且我选择的关系是 OR(例如,Cost IS NULL 或 PurchaseType IS NULL,或者两者都是 NULL)。
开发问题
我的应用程序有几个查询,这些查询以用户进行选择为条件。例如,如果选择字段被命名为 @selection1 和 @selection2 那么我的 where 子句看起来像这样:
select...
from...
where isActive=1
and (@selection1=column or @selection1='')
and (@selection2=column or @selection2='')
这个特殊的用例要求我建立 OR 关系而不是 AND ,这在我的逻辑中增加了一个皱纹:
select..
from...
where isActive=1
and ((@selection1=column or @selection1='')
or (@selection2=column or @selection2=''))
如果任何一个选择为空(每行后半部分为''=''),则整个子句返回 true 并被跳过,返回所有行并忽略用户的有效选择。
代码示例
DECLARE @filter AS VARCHAR(100)=''
/**Options are empty,Cost to show only where Cost IS NULL,PurchaseType to show only where PurchaseType IS NULL,PurchaseQuantity to show only where PurchaseQuantity IS NULL,or any combination of these 3 values to show where ANY COLUMN in the combination IS NULL
so if the user selects 'Cost,PurchaseQuantity',then they want columns 1,2,3,4,7
and if the user selects 'PurchaseType,6.
**/
-- Create Sample Table
IF OBJECT_ID('tempdb..#Product') IS NOT NULL
Truncate TABLE #Product
ELSE
CREATE TABLE #Product (id INT,Cost DEC(8,2),PurchaseType VARCHAR(20),PurchaseQuantity INT,isActive bit);
INSERT INTO #Product
VALUES
(1,NULL,1),(2,10.00,0),(3,'Each',(4,100,(5,5.50,'Box',1,(6,142.00,12,(7,'Pkg',50,1)
-- Select values from Sample Table
-- Works great for single @filter values,but fails for multiple
--SELECT id,Cost,PurchaseType,PurchaseQuantity FROM #Product
--WHERE
-- CASE WHEN @filter = 'Cost' THEN Cost
-- ELSE
-- CASE WHEN @filter='PurchaseType' THEN PurchaseType
-- ELSE
-- CASE WHEN @filter='PurchaseQuantity' THEN PurchaseQuantity
-- ELSE
-- CASE WHEN @filter='Cost,PurchaseType' THEN COALESCE(Cost,PurchaseType)
-- --CASE WHEN @filter='Cost,PurchaseType' THEN Cost or PurchaseType
-- --The above line expresses what I'm trying to do in pseudo,but doesn't work for resulting when Cost IS NULL or PurchaseType IS NULL
-- ELSE
-- CASE WHEN @filter='Cost,PurchaseQuantity' THEN COALESCE(Cost,PurchaseQuantity)
-- ELSE
-- CASE WHEN @filter='PurchaseQuantity,PurchaseType' THEN COALESCE(PurchaseQuantity,PurchaseType)
-- ELSE
-- CASE WHEN @filter='Cost,PurchaseQuantity,PurchaseType)
-- END END END END END END END
-- IS NULL
select id,isActive
from #Product
where isActive=1
and (@filter !='' and
(
(Cost is null and @filter like '%Cost%')
or
(PurchaseType is null and @filter like '%PurchaseType%')
or
(PurchaseQuantity is null and @filter like '%PurchaseQuantity%')
))
我已经添加了@Stu 的逻辑,但是我的生产架构中的“isActive=1 and”和其他逻辑意味着如果用户将过滤器留空而不是忽略 @Filter 部分(这就是我想看到发生)。
如我的代码示例中所述,我最终希望设置 where 子句,以便
- 如果用户没有为@filter 选择任何内容,则跳过 WHERE
- 如果用户为@filter 选择单个值,则 WHERE 返回该单列为 NULL 的位置
- 如果用户为@filter 选择多个值,则 WHERE 返回任何 @filter 列为 NULL 的位置
我已经花了一天半的时间来解决这个问题,但我一无所获……希望我错过了一些简单的东西,并且互联网上的某个人有办法解决这个问题。提前致谢!!
解决方法
我不明白你是怎么想出这个安排的,但这对你有用吗?很难确定!
select id,Cost,PurchaseType,PurchaseQuantity,isActive
from #Product
where isActive=1 and
(@Filter ='' or
(
(Cost is null and @filter like '%cost%')
or
(PurchaseType is null and @filter like '%PurchaseType%')
or
(PurchaseQuantity is null and @filter like '%PurchaseQuantity%')
))
请注意,这几乎肯定会导致查询性能不佳,尤其是在表很大 - 1000 行的情况下。
最好进行 3 个单独的查询并 union
结果 - 这样优化器可以在每个查询上使用合适的索引。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。