如何在sql中优化case表达式

wkftcu5l  于 2021-08-09  发布在  Java
关注(0)|答案(2)|浏览(386)

我需要优化一个长case表达式:

  1. update tf
  2. SET class
  3. case
  4. WHEN tf.column = 'a' THEN (select top 1 from lookup_tbl1 where code = 1)
  5. WHEN tf.column = 'b' THEN (select top 1 from lookup_tbl1 where code = 2)
  6. WHEN tf.column = 'c' THEN (select top 1 from lookup_tbl1 where code = 3)
  7. WHEN tf.column = 'd' THEN (select top 1 from lookup_tbl1 where code = 80)
  8. end
  9. SET name
  10. case
  11. WHEN tf.column2 = 'ryan' THEN (select top 1 from lookup_tbl2 where code = 12)
  12. WHEN tf.column2 = 'david' THEN (select top 1 from lookup_tbl2 where code = 22)
  13. WHEN tf.column2 = 'tan' THEN (select top 1 from lookup_tbl2 where code = 32)
  14. WHEN tf.column2 = 'drake' THEN (select top 1 from lookup_tbl2 where code = 802)
  15. end
  16. SET department
  17. case
  18. WHEN tf.column3 like 'd.d%' then 'Director D'
  19. WHEN tf.column3 like '%AC' then 'Accounting'
  20. end
  21. FROM trans_tbl tf WHERE flag is null

我在清理table。我需要用实际名称替换缩写,并修复该表中的数据。有没有别的办法或者更好的办法?

oprakyz7

oprakyz71#

请注意我的示例数据、解决方案和注解:

  1. -- 1. CREATE TABLES FOR SAMPLE DATA
  2. DECLARE @transform_tbl TABLE
  3. (
  4. col1 CHAR(1),
  5. col2 VARCHAR(20),
  6. class VARCHAR(100),
  7. [name] VARCHAR(100),
  8. flag BIT,
  9. INDEX uq_flag CLUSTERED (flag) -- A Unique nonclustered, filtered index would be best
  10. )
  11. DECLARE @lookup1 TABLE (x INT, xx INT, INDEX idx_lu1 CLUSTERED(x)); -- you want an index on x
  12. DECLARE @lookup2 TABLE (x INT, xx VARCHAR(100), INDEX idx_lu2 CLUSTERED(x)); -- you want an index on x
  13. -- 2. POPULATE TABLES WITH SAMPLE DATA
  14. INSERT @lookup1 VALUES (1,111),(2,222),(3,333),(80,8080);
  15. INSERT @lookup2 VALUES (12,'sally'),(22,'sue'),(32,'betty'),(802,'lulu');
  16. INSERT @transform_tbl(col1,col2,class)
  17. VALUES('a','ryan','xxx'),('b','david','xxx'),('c','tan','xxx'),('d','drake','xxx');
  18. INSERT @transform_tbl(col1,col2,class,flag)
  19. VALUES('x','','xxx',1),('x','','xxx',0),('x','','xxx',1),('x','','xxx',1);
  20. -- 3. SELECT Statement to understand the update
  21. SELECT tf.col1, tf.col2, tf.[name], Class = lu1.XX, [Name] = lu2.XX
  22. FROM @transform_tbl AS tf
  23. CROSS APPLY (VALUES(CASE
  24. WHEN tf.col1 = 'a' THEN 1
  25. WHEN tf.col1 = 'b' THEN 2
  26. WHEN tf.col1 = 'c' THEN 3
  27. WHEN tf.col1 = 'd' THEN 80 END)) AS f(Class)
  28. CROSS APPLY (VALUES(CASE
  29. WHEN tf.col2 = 'ryan' THEN 12
  30. WHEN tf.col2 = 'david' THEN 22
  31. WHEN tf.col2 = 'tan' THEN 32
  32. WHEN tf.col2 = 'drake' THEN 802 END)) AS f2([Name])
  33. CROSS APPLY (SELECT TOP (1) lu.xx from @lookup1 AS lu where x = f.Class) AS lu1(XX)
  34. CROSS APPLY (SELECT TOP (1) lu.xx from @lookup2 AS lu where x = f2.[Name]) AS lu2(XX)
  35. WHERE tf.flag IS NULL;
  36. -- 4. Change the SELECT to an UPDATE
  37. ;
  38. -- SELECT tf.col1, tf.col2, tf.[name], Class = lu1.XX, [Name] = lu2.XX
  39. UPDATE tf SET Class = lu1.XX, [Name] = lu2.XX
  40. FROM @transform_tbl AS tf
  41. CROSS APPLY (VALUES(CASE
  42. WHEN tf.col1 = 'a' THEN 1
  43. WHEN tf.col1 = 'b' THEN 2
  44. WHEN tf.col1 = 'c' THEN 3
  45. WHEN tf.col1 = 'd' THEN 80 END)) AS f(Class)
  46. CROSS APPLY (VALUES(CASE
  47. WHEN tf.col2 = 'ryan' THEN 12
  48. WHEN tf.col2 = 'david' THEN 22
  49. WHEN tf.col2 = 'tan' THEN 32
  50. WHEN tf.col2 = 'drake' THEN 802 END)) AS f2([Name])
  51. CROSS APPLY (SELECT TOP (1) lu.xx from @lookup1 AS lu where x = f.Class) AS lu1(XX)
  52. CROSS APPLY (SELECT TOP (1) lu.xx from @lookup2 AS lu where x = f2.[Name]) AS lu2(XX)
  53. WHERE tf.flag IS NULL
  54. -- 5. Review Results
  55. SELECT tf.col1, tf.[name], tf.class
  56. FROM @transform_tbl AS tf
  57. WHERE tf.flag IS NULL;

结果(之前):

  1. col1 col2 name Class Name
  2. ---- -------------------- ------ ------- ---------
  3. a ryan NULL 111 sally
  4. b david NULL 222 sue
  5. c tan NULL 333 betty
  6. d drake NULL 8080 lulu

结果(之后):

  1. col1 name class
  2. ---- ------- --------
  3. a sally 111
  4. b sue 222
  5. c betty 333
  6. d lulu 8080

如果您了解apply的工作原理,那么这段代码非常干净,并且易于维护。
注意我使用的索引。生成的执行计划将是所有查找和嵌套循环联接,这是对于这样的查询(使用标准的、常规的方法)所能获得的最佳结果

展开查看全部
xa9qqrwz

xa9qqrwz2#

对于每个记录,一个完整的select将启动它(因此,如果在update表中有1000条记录,您将执行其他1000条select)。优化这个查询的一个简单方法是将select保存到变量中。

  1. declare @a varchar(20), -- I don't know the type..
  2. @b varchar(20), ..
  3. @ryan varchar(20), ..
  4. set @a = select top 1 from lookup_tbl1 where code = 1
  5. set @b = select top 1 from lookup_tbl1 where code = 1
  6. ...
  7. update tf
  8. SET class
  9. case
  10. WHEN tf.column = 'a' THEN @a
  11. WHEN tf.column = 'b' THEN @b
  12. WHEN tf.column = 'c' THEN @c
  13. WHEN tf.column = 'd' THEN @d
  14. end
  15. SET name
  16. case
  17. WHEN tf.column2 = 'ryan' THEN @ryan
  18. WHEN tf.column2 = 'david' THEN @david
  19. WHEN tf.column2 = 'tan' THEN @tan
  20. WHEN tf.column2 = 'drake' THEN @drake
  21. end
  22. SET department
  23. case
  24. WHEN tf.column3 like 'd.d%' then 'Director D'
  25. WHEN tf.column3 like '%AC' then 'Accounting'
  26. end
  27. FROM transform_tbl tf WHERE flag is null

希望这对你有帮助

展开查看全部

相关问题