如何使用尽可能少的查询(一个?)

How to build hierarchy using as few queries as possible (one?)

我有一个较新的 SQL 服务器,其数据库包含一个 table 客户和一个 table 关系(parent/child 关系)。这些关系可能有不同的类型(即标准公司层次结构、朋友、竞争等)。

我想从单个客户开始列出完整的公司层次结构(特定关系类型)。只是一个完整的列表,它不必订购,因为它会在外面处理。

我可以用我的 web-script 来构建视觉部分,但那里可能有一些巨大的层次结构,这将导致对 运行 的数百个单一查询。那可能会变得太time-consuming.

我试着看一下 CTE(常见的 table 表达式),但我认为我并不真正理解它。我似乎能够列出我的初始客户所在的所有地方,但根本不属于它的地方。

表格:

CREATE TABLE CUSTOMERS
(
 id int not null, 
 name varchar(100) not null
);


CREATE TABLE RELATIONSHIPS
(
 relationid int not null,
 relationtype int not null,
 customerid int not null,
 parentid int not null
);

INSERT INTO CUSTOMERS(id,name) VALUES 
(1040,'Cust A'),
(1041,'Cust B'),
(1042,'Cust C'),
(1043,'Cust D'),
(1044,'Cust E'),
(1045,'Cust F'),
(1046,'Cust G'),
(1047,'Cust H');

INSERT INTO RELATIONSHIPS(relationid,relationtype,customerid,parentid)
VALUES
(1,1,1041,1040),
(2,1,1042,1040),
(3,1,1043,1042),
(4,1,1047,1043),
(5,2,1041,1040);

当前SQL

with cte as (
    select t.parentid,parent.name as parentname,t.customerid,child.name as childname
    from RELATIONSHIPS t
    INNER JOIN CUSTOMERS parent ON t.parentid = parent.id
    INNER JOIN CUSTOMERS child ON t.customerid = child.id
    where relationtype = 1 and (customerid = 1042 or parentid = 1042)

    union all

    select t.parentid,parent.name as parentname,t.customerid,child.name as childname
    from RELATIONSHIPS t
    INNER JOIN CUSTOMERS parent ON t.parentid = parent.id
    INNER JOIN CUSTOMERS child ON t.customerid = child.id
    inner join cte c on (c.customerid=t.parentid )
    where relationtype = 1
)
select distinct t.* from cte t;

Fiddle: http://sqlfiddle.com/#!18/3e919b/14

根据上面数据中的数据,我要列出这个:

1040,Cust A,1041,Cust B
1040,Cust A,1042,Cust C
1042,Cust C,1041,Cust D
1043,Cust D,1047,Cust H

我上面的查询尝试列出客户 1042 ( Cust C) 所在的整个层次结构树。通过我的查询,我似乎得到了所有,除了 child 1041 (Cust B) 的查询。不确定如何在查询中包含此类内容,因为它与我的起始公司没有直接关系(除了在同一层次结构中)。

天哪,这太丑了。不知道这是否是最佳表演者。您实际上是在说,您想要客户 ('Cust C') 的所有 children、客户的所有 parents、 然后所有那些 parents 中的 children(不在先前的分支中)。我最终用 3 个 rCTE 完成了这项工作,这在更大的数据集上可能是绝对的性能杀手,但是嘿,它得到了工作 "done":

USE Sandbox;
GO

CREATE TABLE CUSTOMERS
(
 id int not null, 
 name varchar(100) not null
);


CREATE TABLE RELATIONSHIPS
(
 relationid int not null,
 relationtype int not null,
 customerid int not null,
 parentid int not null
);

INSERT INTO CUSTOMERS(id,name) VALUES 
(1040,'Cust A'),
(1041,'Cust B'),
(1042,'Cust C'),
(1043,'Cust D'),
(1044,'Cust E'),
(1045,'Cust F'),
(1046,'Cust G'),
(1047,'Cust H');

INSERT INTO RELATIONSHIPS(relationid,relationtype,customerid,parentid)
VALUES
(1,1,1041,1040),
(2,1,1042,1040),
(3,1,1043,1042),
(4,1,1047,1043),
(5,2,1041,1040);
GO

DECLARE @Customer varchar(100) = 'Cust C';
--Get the children of the Customer
WITH Children AS(
    SELECT Cp.name AS ParentName,
           Cp.Id AS ParentID,
           CC.name AS ChildName,
           Cc.id AS ChildID
    FROM CUSTOMERS Cp
         JOIN RELATIONSHIPS R ON Cp.id = R.parentid
         JOIN CUSTOMERS Cc ON R.customerid = Cc.id
    WHERE Cp.name = @Customer
    UNION ALL
    SELECT C.ChildName AS ParentName,
           C.ChildID AS ParentID,
           Cc.name AS ChildName,
           Cc.Id AS ChildId
    FROM Children C
         JOIN RELATIONSHIPS R ON C.ChildID = r.parentid
         JOIN CUSTOMERS Cc ON R.customerid = Cc.id),
--Get the Parents of the customer
Parents AS(
    SELECT Cp.name AS ParentName,
           Cp.Id AS ParentID,
           CC.name AS ChildName,
           Cc.id AS ChildID
    FROM CUSTOMERS Cc
         JOIN RELATIONSHIPS R ON Cc.id = R.customerid
         JOIN CUSTOMERS Cp ON R.parentid = Cp.id
    WHERE Cc.name = @Customer
    UNION ALL
    SELECT Cp.name AS ParentName,
           Cp.Id AS ParentID,
           P.ParentName AS ChildName,
           P.ParentID AS ChildId
    FROM Parents P
         JOIN RELATIONSHIPS R ON P.ParentID = R.customerid
         JOIN CUSTOMERS Cp ON R.parentid = Cp.id),
--Get the children of the parents. Yuck
ParentChildren AS(
    SELECT DISTINCT
           P.ParentName AS ParentName,
           P.ParentID AS ParentID,
           Cc.name AS ChildName,
           Cc.id AS ChildId
    FROM Parents P
         JOIN RELATIONSHIPS R ON P.ParentID = R.parentid
         JOIN CUSTOMERS Cc ON R.customerid = Cc.id    
    WHERE NOT EXISTS (SELECT 1
                      FROM Parents E
                      WHERE E.ChildID = Cc.id
                        AND E.ParentID = P.ParentID)
    UNION ALL    
    SELECT PC.ChildName AS ParentName,
           PC.ChildId AS ParentID,
           Cc.name AS ChildName,
           Cc.id AS ChildID
    FROM ParentChildren PC
         JOIN RELATIONSHIPS R ON PC.ChildId = R.parentid
         JOIN CUSTOMERS Cc ON R.customerid = Cc.id
    WHERE NOT EXISTS (SELECT 1
                      FROM Parents E
                      WHERE E.ChildID = Cc.id
                        AND E.ParentID = PC.ParentID)
)

SELECT *
FROM Children
UNION ALL
SELECT *
FROM Parents
UNION ALL
SELECT *
FROM ParentChildren
ORDER BY ParentID ASC;

GO

DROP TABLE RELATIONSHIPS;
DROP TABLE CUSTOMERS;