如何让 SQL 重复一些基于集合的操作任意次数而不循环?如何让 SQL 对一系列数字执行操作?我基本上是在寻找一种方法来执行基于集合的 for 循环。
我知道我可以创建一个包含整数的小表,比如从 1 到 1000,然后将其用于该范围内的范围操作。
例如,如果我有那个表,我可以像这样进行选择以查找数字 100-200 的总和:
select sum(n) from numbers where n between 100 and 200
有什么想法吗?我有点在寻找适用于 T-SQL 的东西,但任何平台都可以。
[编辑] 我有自己的解决方案,使用 SQL CLR,它适用于 MS SQL 2005 或 2008。See below.
最佳答案
我认为对您问题的简短回答是使用 WITH 子句生成您自己的子句。
不幸的是,数据库中的大牌并没有内置的可查询数字范围伪表。或者,更一般地说,简单的纯 SQL 数据生成功能。就我个人而言,我认为这是一个巨大的失败,因为如果他们这样做了,就有可能移动当前锁定在过程脚本(T-SQL、PL/SQL 等)中的大量代码。 ) 转换为纯 SQL,这对性能和代码复杂性有许多好处。
所以无论如何,从一般意义上来说,您需要的是动态生成数据的能力。
Oracle 和 T-SQL 都支持可用于执行此操作的 WITH 子句。它们在不同的 DBMS 中的工作方式略有不同,MS 称它们为“通用表表达式”,但它们在形式上非常相似。将这些与递归结合使用,您可以相当轻松地生成一系列数字或文本值。这是它可能的样子......
在 Oracle SQL 中:
WITH
digits AS -- Limit recursion by just using it for digits.
(SELECT
LEVEL - 1 AS num
FROM
DUAL
WHERE
LEVEL < 10
CONNECT BY
num = (PRIOR num) + 1),
numrange AS
(SELECT
ones.num
+ (tens.num * 10)
+ (hundreds.num * 100)
AS num
FROM
digits ones
CROSS JOIN
digits tens
CROSS JOIN
digits hundreds
WHERE
hundreds.num in (1, 2)) -- Use the WHERE clause to restrict each digit as needed.
SELECT
-- Some columns and operations
FROM
numrange
-- Join to other data if needed
这无疑是相当冗长的。 Oracle 的递归功能是有限的。语法很笨拙,性能不佳,并且仅限于 500 个(我认为)嵌套级别。这就是为什么我选择只对前 10 位数字使用递归,然后交叉(笛卡尔)连接将它们组合成实际数字。
我自己没有使用过 SQL Server 的公用表表达式,但由于它们允许自引用,因此递归比 Oracle 中的要简单得多。性能是否具有可比性,以及嵌套限制是什么,我不知道。
无论如何,递归和 WITH 子句在创建需要即时生成的数据集的查询时都是非常有用的工具。然后通过查询这个数据集,对值进行操作,就可以得到各种不同类型的生成数据。聚合、复制、组合、排列等等。您甚至可以使用此类生成的数据来帮助汇总或深入了解其他数据。
更新:我只想补充一点,一旦您开始以这种方式处理数据,您就会对 SQL 的新思维方式敞开心扉。它不仅仅是一种脚本语言。这是一个相当强大的数据驱动 declarative language .有时使用起来很痛苦,因为多年来它一直缺乏增强功能来帮助减少复杂操作所需的冗余。但尽管如此,它还是非常强大,并且是一种将数据集作为算法的目标和驱动程序的相当直观的方式。
关于基于 SQL 集的范围,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/58429/