zoukankan      html  css  js  c++  java
  • 在sql server中利用with as实现递归功能

    在sqlserver2005之前,要实现递归功能比较麻烦,比如可能会要用到临时表与while语句来循环。自sqlserver2005之后,新增了with as功能语法,即 公用表达式(CTE),让递归实现起来变的简单了。

    本章我们主要演示如何利用with as功能实现一个简单的递归功能。
    在这之前先看一下cte的语法:

    [ WITH <common_table_expression > [ ,...n ] ]
    <common_table_expression >::=
    expression_name [ ( column_name [ ,...n ] ) ]
    AS
    ( CTE_query_definition )
    参数说明:
    expression_name:
    公用表表达式的有效标识符。 expression_name 必须与在同一 WITH <common_table_expression > 子句中定义的任何其他公用表表达式的名称不同,但 expression_name 可以与基表或基视图的名称相同。在查询中对 expression_name 的任何引用都会使用公用表表达式,而不使用基对象。

    column_name:
    在公用表表达式中指定列名。在一个 CTE 定义中不允许出现重复的名称。指定的列名数必须与CTE_query_definition结果集中列数匹配。只有在查询定义中为所有结果列都提供了不同的名称时,列名称列表才是可选的。

    CTE_query_definition:
    指定一个其结果集填充公用表表达式的 SELECT 语句。除了 CTE 不能定义另一个 CTE 以外,CTE_query_definition的 SELECT 语句必须满足与创建视图时相同的要求。
    如果定义了多个 CTE_query_definition,则这些查询定义必须用下列一个集合运算符联接起来:UNION ALL、UNION、EXCEPT 或 INTERSECT。

    --开始实例演示--

    先创建一个仓库表,表名为Storage_Depository,该表有三个字段:DID(仓库编号),DName(仓库名称),PID(父仓库编号).
    通过这样一个简单表,就可以将所有仓库信息,通过DID与PID字段来创建一个树型结构。
    创建表的sql语句:

    Create table Storage_Depository
    (
    DID varchar(50) not null primary key,
    DName varchar(50) not null,
    PID varchar(50) null
    )
    然后往该表插入演示数据:

    insert into Storage_Depository(DID,DName,PID)
    select 'A','A仓库',null
    union all
    select 'A-1','A-1仓库','A'
    union all
    select 'A-2','A-2仓库','A'
    union all
    select 'A-1-1','A-1-1仓库','A-1'
    union all
    select 'B','B仓库',null
    从上面的数据可以看的出来,A的子仓为A-1与A-2仓,而A-1-1为A-1的子仓,B仓是一个独立的仓库,与A仓平级。
    下面,我们通过with as功能,查出A仓下面的所有子仓:

    with w_Storage_Depository as
    (
    select DID,DName,PID from Storage_Depository where DID='A'
    union all
    select A.DID,A.DName,A.PID from Storage_Depository A,w_Storage_Depository B where A.PID=B.DID
    )
    select * from w_Storage_Depository
    代码很简短,也非常容易让人理解.

    反过来,比如我们要查出A-1-1仓的所有上级仓,稍稍改一下上面的sql语句就可以了:

    with w_Storage_Depository as
    (
    select DID,DName,PID from Storage_Depository where DID='A-1-1'
    union all
    select A.DID,A.DName,A.PID from Storage_Depository A,
    w_Storage_Depository B where A.DID=B.PID
    )
    select * from w_Storage_Depository
    很简单吧,熟练使用CTE后,会发现它会给我们的工作带来很大的便利.

  • 相关阅读:
    A/B-test
    LDA概率主题模型
    减肥经验帖
    wide and deep
    利用责任链模式设计一个拦截器
    分享几个 SpringBoot 实用的小技巧
    「造个轮子」——cicada 设计一个配置模块
    「造个轮子」——cicada 源码分析
    「造个轮子」——cicada(轻量级 WEB 框架)
    强如 Disruptor 也发生内存溢出?
  • 原文地址:https://www.cnblogs.com/lgx5/p/6040940.html
Copyright © 2011-2022 走看看