INNER JOIN with ON 除一列外的所有列
INNER JOIN with ON All columns except one Column
我有 2 个 tables(表 1 和表 2)。两个 table 的架构完全相同,并且都可能有重复的记录集,除了 ID,因为 ID 是自动生成的。
我想获取通用记录集,但 ID 跟在表 1 的 ID 后面。因此,我使用内部联接进行查询。它按我的预期工作。
SELECT Table1.ID, Table1.Param1, Table1.Param2, Table1.Param3
INTO #Common
FROM Table1
INNER JOIN Table2 ON Table1.Param1 = Table2.Param1
AND Table1.Param2 = Table2.Param2
AND Table1.Param3 = Table2.Param3
但是在实际使用中,两个table的参数总数会在100个左右,所以ON子句中的比较总数会增加到100个。
如何通过排除一列而不是比较 ON 子句中的所有列来进行内部联接?
通过从两个 table 中删除 ID 列并进行相交也是不可能的,因为我仍然想提取 Table1 ID 用于其他目的。
我可以通过删除 ID 并比较那些 2 table 来实现 2 table 的共同点。
但是,这仍然不能满足我的要求,因为我需要为这些公共数据获取 Table1 ID。
SELECT * INTO #TemporaryTable1 FROM Table1
ALTER TABLE #TemporaryTable1 DROP COLUMN ID
SELECT * INTO #TemporaryTable2 FROM Table2
ALTER TABLE #TemporaryTable2 DROP COLUMN ID
SELECT * INTO #Common FROM (SELECT * FROM #TemporaryTable1 INTERSECT SELECT * FROM #TemporaryTable2) data
SELECT * FROM #Common
如果我正确理解了你的问题,我想你可以使用以下代码动态生成你想要使用的查询:
DECLARE @SQL nvarchar(max) = 'SELECT ',
@TBL1 nvarchar(50) = 'data',
@TBL2 nvarchar(50) = 'data1',
@EXCLUDEDCOLUMNS nvarchar(100)= 'ID,col1'
-- column selection
SELECT @sql += @tbl1 + '.' + COLUMN_NAME + ' ,
'
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_NAME = @TBL1
-- from clause and remove last ,
set @SQL = LEFT(@sql,LEN(@sql) - 5)
SET @sql += '
FROM ' + @TBL1 + ' INNER JOIN
' + @TBL2 + '
ON '
-- define the on clause
SELECt @SQL += @tbl1 + '.' + COLUMN_NAME + ' = '+ @tbl2 + '.' + COLUMN_NAME +',
'
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_NAME = @TBL1
AND COLUMN_NAME not in (@EXCLUDEDCOLUMNS)
--remove last ,
set @SQL = LEFT(@sql,LEN(@sql) - 3)
--SELECt @SQL
EXEC SP_EXECUTESQL @sql
在执行之前确保正确生成了@sql。使用 @EXCLUDEDCOLUMNS 参数选择您要从 on 子句中排除的列。
我有 2 个 tables(表 1 和表 2)。两个 table 的架构完全相同,并且都可能有重复的记录集,除了 ID,因为 ID 是自动生成的。
我想获取通用记录集,但 ID 跟在表 1 的 ID 后面。因此,我使用内部联接进行查询。它按我的预期工作。
SELECT Table1.ID, Table1.Param1, Table1.Param2, Table1.Param3
INTO #Common
FROM Table1
INNER JOIN Table2 ON Table1.Param1 = Table2.Param1
AND Table1.Param2 = Table2.Param2
AND Table1.Param3 = Table2.Param3
但是在实际使用中,两个table的参数总数会在100个左右,所以ON子句中的比较总数会增加到100个。 如何通过排除一列而不是比较 ON 子句中的所有列来进行内部联接?
通过从两个 table 中删除 ID 列并进行相交也是不可能的,因为我仍然想提取 Table1 ID 用于其他目的。 我可以通过删除 ID 并比较那些 2 table 来实现 2 table 的共同点。 但是,这仍然不能满足我的要求,因为我需要为这些公共数据获取 Table1 ID。
SELECT * INTO #TemporaryTable1 FROM Table1
ALTER TABLE #TemporaryTable1 DROP COLUMN ID
SELECT * INTO #TemporaryTable2 FROM Table2
ALTER TABLE #TemporaryTable2 DROP COLUMN ID
SELECT * INTO #Common FROM (SELECT * FROM #TemporaryTable1 INTERSECT SELECT * FROM #TemporaryTable2) data
SELECT * FROM #Common
如果我正确理解了你的问题,我想你可以使用以下代码动态生成你想要使用的查询:
DECLARE @SQL nvarchar(max) = 'SELECT ',
@TBL1 nvarchar(50) = 'data',
@TBL2 nvarchar(50) = 'data1',
@EXCLUDEDCOLUMNS nvarchar(100)= 'ID,col1'
-- column selection
SELECT @sql += @tbl1 + '.' + COLUMN_NAME + ' ,
'
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_NAME = @TBL1
-- from clause and remove last ,
set @SQL = LEFT(@sql,LEN(@sql) - 5)
SET @sql += '
FROM ' + @TBL1 + ' INNER JOIN
' + @TBL2 + '
ON '
-- define the on clause
SELECt @SQL += @tbl1 + '.' + COLUMN_NAME + ' = '+ @tbl2 + '.' + COLUMN_NAME +',
'
FROM INFORMATION_SCHEMA.COLUMNS
WHERE TABLE_NAME = @TBL1
AND COLUMN_NAME not in (@EXCLUDEDCOLUMNS)
--remove last ,
set @SQL = LEFT(@sql,LEN(@sql) - 3)
--SELECt @SQL
EXEC SP_EXECUTESQL @sql
在执行之前确保正确生成了@sql。使用 @EXCLUDEDCOLUMNS 参数选择您要从 on 子句中排除的列。