使用SQL&;连接多个表;T-SQL
不幸的是,我不能确定我的问题的名字在这里是正确的 初始数据示例:使用SQL&;连接多个表;T-SQL,sql,sql-server,tsql,sql-server-2017,Sql,Sql Server,Tsql,Sql Server 2017,不幸的是,我不能确定我的问题的名字在这里是正确的 初始数据示例: Table 1 Table 2 Table 3 | ID | Name | | ID | Info1 | | ID | Info2 | |----|-------| |----|-------| |----|-------| | 1 | Name1 | | 1 | text1 | | 1 | text1 | | 2 | Name2 | |
Table 1 Table 2 Table 3
| ID | Name | | ID | Info1 | | ID | Info2 |
|----|-------| |----|-------| |----|-------|
| 1 | Name1 | | 1 | text1 | | 1 | text1 |
| 2 | Name2 | | 1 | text1 | | 1 | text1 |
| 3 | Name3 | | 2 | text2 | | 1 | text1 |
| 2 | text2 | | 2 | text2 |
| 3 | text3 |
在我的初始数据中,我按字段ID
在3个表之间建立了关系
我需要将table2
和table3
连接到第一个表,但是如果我进行顺序连接,比如left join table2
和left join table3
通过ID
进行连接,我将在第二次连接时获得额外的记录,因为在第一次连接后将有多个记录具有一个ID
我需要获取table2
和table3
的记录,就像第一个表的ID在每列中的列表一样
下面是一个预期结果的示例:
Table 3
| ID | Name |Info1(Table2)|Info2(Table3)|
|-------|-----------|-------------|-------------|
| 1 | Name1 | text1 | text1 |
| 1 | Name1 | text1 | text1 |
| 1 | Name1 | null | text1 |
| 2 | Name2 | text2 | text2 |
| 2 | Name2 | text2 | null |
| 3 | Name3 | null | text3 |
这是我将使用的方法,但是,您的表格设计可能会有所改进;首先,为什么表2和表3是分开的
USE Sandbox;
GO
CREATE TABLE dbo.Table1 (ID int, [Name] varchar(5))
INSERT INTO dbo.Table1 (ID,
[Name])
VALUES(1,'Name1'),
(2,'Name1'),
(3,'Name3');
CREATE TABLE dbo.Table2 (Id int,Info1 varchar(5));
CREATE TABLE dbo.Table3 (Id int,Info2 varchar(5));
INSERT INTO dbo.Table2 (Id,
Info1)
VALUES(1,'text1'),
(1,'text1'),
(2,'text2'),
(2,'text2');
INSERT INTO dbo.Table3 (Id,
Info2)
VALUES(1,'text1'),
(1,'text1'),
(1,'text1'),
(2,'text2'),
(3,'text3');
WITH T2 AS(
SELECT ID,
Info1,
ROW_NUMBER() OVER (PARTITION BY ID ORDER BY (SELECT NULL)) AS RN --SELECT NULL as you have no other columns to actually create an order
FROM Table2),
T3 AS(
SELECT ID,
Info2,
ROW_NUMBER() OVER (PARTITION BY ID ORDER BY (SELECT NULL)) AS RN
FROM Table3),
Tally AS(
SELECT ROW_NUMBER() OVER (ORDER BY (SELECT NULL)) AS I --Assuming you have 10 or less matching items
FROM (VALUES(NULL),(NULL),(NULL),(NULL),(NULL),(NULL),(NULL),(NULL),(NULL),(NULL)) N(N))
SELECT T1.ID,
T1.[Name],
T2.info1,
T3.info2
FROM Table1 T1
CROSS JOIN Tally T
LEFT JOIN T2 ON T1.ID = T2.ID AND T.I = T2.RN
LEFT JOIN T3 ON T1.ID = T3.ID AND T.I = T3.RN
WHERE T2.ID IS NOT NULL OR T3.ID IS NOT NULL
ORDER BY T1.ID, T.I;
GO
DROP TABLE dbo.Table1;
DROP TABLE dbo.Table2;
DROP TABLE dbo.Table3;
如果您有超过10行,那么您可以动态构建一个“适当的”理货表,或者创建一个物理理货表。不过,一个正在运行的可能会是一个更好的主意,因为我怀疑您是否会有100个匹配行。您是否尝试过内部连接?@TimBiegeleisen修复了我的问题,将
T.I=T2.RN
保留在外(在我创建理货表之后,这是非常愚蠢的)