基于SQL集的范围

Haf*_*hor 3 sql sql-server

如何在没有循环的情况下让SQL重复一些基于集合的操作任意次?如何让SQL对一系列数字执行操作?我基本上是在寻找一种基于集合的for循环的方法.

我知道我可以创建一个包含整数的小表,比如从1到1000,然后将其用于该范围内的范围操作.

例如,如果我有那个表,我可以做一个选择,找到100-200这样的数字总和:

select sum(n) from numbers where n between 100 and 200
Run Code Online (Sandbox Code Playgroud)

有任何想法吗?我有点想找一些适合T-SQL的东西,但任何平台都没关系.

[编辑]我有自己的解决方案,使用SQL CLR,适用于MS SQL 2005或2008. 请参阅下文.

Chr*_*man 5

我认为你的问题的答案非常简短,就是使用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
Run Code Online (Sandbox Code Playgroud)

这无疑是相当冗长的.Oracle的递归功能有限.语法很笨拙,性能不高,而且只限于500(我认为)嵌套级别.这就是为什么我选择仅对前10个数字使用递归,然后交叉(笛卡尔)连接以将它们组合成实际数字.

我自己没有使用过SQL Server的Common Table Expressions,但由于它们允许自引用,因此递归比Oracle更简单.无论性能是否具有可比性,以及嵌套限制是什么,我都不知道.

无论如何,递归和WITH子句是创建需要动态生成的数据集的查询的非常有用的工具.然后通过查询此数据集,对值进行操作,您可以获得各种不同类型的生成数据.聚合,复制,组合,排列等.您甚至可以使用此类生成的数据来帮助汇总或深入查看其他数据.

更新:我只想补充一点,一旦你开始以这种方式处理数据,它就会开始思考SQL的新思路.它不仅仅是一种脚本语言.它是一种相当强大的数据驱动的声明性语言.有时使用会很痛苦,因为多年来它一直缺乏增强功能,有助于减少复杂操作所需的冗余.但是,它非常强大,并且是一种非常直观的方式,可以将数据集作为算法的目标和驱动程序.