如何使用尽可能少的查询(一个?)
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;
我有一个较新的 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;