SQL Server 怎样生成序列号(虚拟数字辅助表)

</pre><pre name="code" class="sql">--生成一个"序列" 或者说生成一个"虚拟数字辅助表"是扩展数据库集合操作的重要途径。其主要创建途径一般是通过笛卡尔积法、系统表法(实质一般也是笛卡尔积法)、创建自定义函数法等方式。  
  
--笛卡尔积法,先构建一个包含10行记录的辅助表,选择10行记录主要是为了方便交叉连接时按照10进制扩展行数  
WITH t1 AS (  
SELECT 1 AS RN  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
UNION All  
SELECT 1  
)  
SELECT NUM=ROW_NUMBER() OVER(ORDER BY a.RN) FROM T1 a,T1 b,T1 c,T1 d,T1 e,T1 f,T1 g  
--生成e10+8(一千万)耗时01:12  
  
--递归法,递归法另一用途是层次查询遍历各级节点;在实现各种复杂数学序列中亦有应用  
--注意当递归次数过百时需加OPTION(MAXRECURSION 0)控制条件使数据库不限定递归次数  
WITH T1 AS (  
SELECT 1 AS NUM  
UNION ALL  
SELECT T1.NUM+1  
FROM T1  
WHERE T1.NUM<10000000)  
SELECT NUM FROM T1  
OPTION(MAXRECURSION 0)  
--生成e10+8(一千万)耗时02:45  
  
--系统表法,生成0-2047  
SELECT number FROM master..spt_values WHERE type='p'  
  
SELECT TOP 10000000 NUM=ROW_NUMBER() OVER(ORDER BY GETDATE())     
FROM syscolumns a,syscolumns b,syscolumns c  
--生成e10+8(一千万)耗时01:16  
  
--自定义函数法  
-- definition of GetNums function, SQL Server 2012 version  
--注意生成的"虚拟数字辅助表"的列别名是"n"  
USE TSQL2012;  
IF OBJECT_ID('dbo.GetNums', 'IF') IS NOT NULL DROP FUNCTION dbo.GetNums;  
GO  
CREATE FUNCTION dbo.GetNums(@low AS BIGINT, @high AS BIGINT) RETURNS TABLE  
AS  
RETURN  
  WITH  
    L0   AS (SELECT c FROM (VALUES(1),(1)) AS D(c)),  
    L1   AS (SELECT 1 AS c FROM L0 AS A CROSS JOIN L0 AS B),  
    L2   AS (SELECT 1 AS c FROM L1 AS A CROSS JOIN L1 AS B),  
    L3   AS (SELECT 1 AS c FROM L2 AS A CROSS JOIN L2 AS B),  
    L4   AS (SELECT 1 AS c FROM L3 AS A CROSS JOIN L3 AS B),  
    L5   AS (SELECT 1 AS c FROM L4 AS A CROSS JOIN L4 AS B),  
    Nums AS (SELECT ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) AS rownum  
            FROM L5)  
  SELECT @low + rownum - 1 AS n  
  FROM Nums  
  ORDER BY rownum  
  OFFSET 0 ROWS FETCH FIRST @high - @low + 1 ROWS ONLY;  
GO  
  
-- definition of GetNums function, pre-SQL Server 2012 version  
IF OBJECT_ID('dbo.GetNums', 'IF') IS NOT NULL  
  DROP FUNCTION dbo.GetNums;  
GO  
CREATE FUNCTION dbo.GetNums(@low AS BIGINT, @high AS BIGINT) RETURNS TABLE  
AS  
RETURN  
  WITH  
    L0   AS (SELECT c FROM (VALUES(1),(1)) AS D(c)),  
    L1   AS (SELECT 1 AS c FROM L0 AS A CROSS JOIN L0 AS B),  
    L2   AS (SELECT 1 AS c FROM L1 AS A CROSS JOIN L1 AS B),  
    L3   AS (SELECT 1 AS c FROM L2 AS A CROSS JOIN L2 AS B),  
    L4   AS (SELECT 1 AS c FROM L3 AS A CROSS JOIN L3 AS B),  
    L5   AS (SELECT 1 AS c FROM L4 AS A CROSS JOIN L4 AS B),  
    Nums AS (SELECT ROW_NUMBER() OVER(ORDER BY (SELECT NULL)) AS rownum  
            FROM L5)  
  SELECT TOP(@high - @low + 1) @low + rownum - 1 AS n  
  FROM Nums  
  ORDER BY rownum;  
GO  
  
-- test function  
--生成e10+8(一千万)耗时01:18  
SELECT n FROM dbo.GetNums(1, 10000000);  
  
--生成11至20的序列  
SELECT n FROM dbo.GetNums(11, 20);  
  
--用函数生成日期序列,注意日期序列可由递归法直接生成,但日期类型必须是datetime,不能是date类型  
DECLARE   
  @start AS DATE = '20120201',  
  @end   AS DATE = '20120212';  
SELECT DATEADD(day, n, @start) AS dt  
FROM dbo.GetNums(0, DATEDIFF(day, @start, @end)) AS Nums;  
GO  
  
DECLARE   
  @start AS DATETIME2 = '2012-02-12 00:00:00.0000000',  
  @end   AS DATETIME2 = '2012-02-18 12:00:00.0000000';  
SELECT DATEADD(hour, n*12, @start) AS dt  
FROM dbo.GetNums(0, DATEDIFF(hour, @start, @end)/12) AS Nums;  
GO  
[sql] view plain copy
--T-SQL  
CREATE TABLE #NumberSequense(RN INT);  
DECLARE @i int;   
SET @i = 1  
  
WHILE @i <= 1000  
BEGIN   
INSERT INTO #NumberSequense  
SELECT @i;  
SET @i = @i + 1  
END  
  
SELECT *   
  FROM #NumberSequense  
 ORDER BY RN  
DROP TABLE #NumberSequense;  
--T-SQL  
CREATE TABLE #NumberSequense(RN INT);  
DECLARE @i int;   
SET @i = 1  
  
WHILE @i <= 1000  
BEGIN   
INSERT INTO #NumberSequense  
SELECT @i;  
SET @i = @i + 1  
END  
  
SELECT *   
  FROM #NumberSequense  
 ORDER BY RN  
DROP TABLE #NumberSequense;  

 

posted @ 2017-11-24 16:09  易小川  阅读(3461)  评论(0编辑  收藏  举报