SQL เปรียบเทียบข้อมูลจากสองตาราง


90

ฉันมี 2 โต๊ะTableAและTableBที่มีรูปแบบเดียวกันของคอลัมน์เช่นตารางทั้งสองTableAและTableBมีคอลัมน์

A B C D E F 

โดยที่ A และ B เป็นคีย์หลัก

วิธีการเขียน SQL เพื่อตรวจสอบว่าถ้าTableAและTableBที่มีคีย์หลักเดียวกันมีค่าเดียวกันในทุกคอลัมน์หรือไม่

หมายความว่าสองตารางนี้มีข้อมูลที่เหมือนกันทุกประการ

คำตอบ:


82

คุณควรจะสามารถ "MINUS" หรือ "EXCEPT" ได้ขึ้นอยู่กับรสชาติของ SQL ที่ DBMS ของคุณใช้

select * from tableA
minus
select * from tableB

หากคิวรีไม่ส่งคืนแถวข้อมูลจะเหมือนกันทุกประการ


5
ข้อเสนอแนะที่ยอดเยี่ยม อย่างไรก็ตามฉันคิดว่าสิ่งนี้อาจไม่ได้ผลหาก tableB มีแถวพิเศษดังนั้นคุณอาจต้องการเปรียบเทียบจำนวนแถวเพิ่มเติม
jzd

5
วิธีอื่น ๆ จะไม่ทำงานหากtableAมีแถวพิเศษ คุณคงต้องการ(A EXCEPT B) INTERSECT (B EXCEPT A)ฉันเดาว่านี่จะมีประสิทธิภาพน้อยกว่าการเข้าร่วมมาตรฐานที่ลุ่มมาก
Martin Smith

แบบสอบถามส่งคืนผลลัพธ์สองชุด ??
BuZz

แบบสอบถามนี้จะส่งคืนแถวด้วย NULLS ถ้ามี
Reeya Oberoi

6
@ แฟรงคลิน - ไม่ควรส่งคืนผลลัพธ์เพียงชุดเดียว หากคุณได้รับสองให้ใช้ EXCEPT แทน MINUS
MTS

57

การใช้ตัวดำเนินการเชิงสัมพันธ์:

SELECT * FROM TableA
UNION 
SELECT * FROM TableB
EXCEPT 
SELECT * FROM TableA
INTERSECT
SELECT * FROM TableB;

เปลี่ยนEXCEPTเป็นMINUSสำหรับ Oracle

จุดที่จู้จี้จุกจิกเล็กน้อย: ข้างต้นขึ้นอยู่กับลำดับความสำคัญของตัวดำเนินการซึ่งตามมาตรฐาน SQL นั้นขึ้นอยู่กับการใช้งานดังนั้น YMMV ใช้งานได้กับ SQL Server ซึ่งมีความสำคัญกว่า:

  1. นิพจน์ในวงเล็บ
  2. INTERSECT
  3. EXCEPTและUNIONประเมินจากซ้ายไปขวา

สำหรับ Oracle คุณต้องใช้วงเล็บรอบ UNION วงเล็บรอบ ๆ INTERSECT และ (ตามที่ระบุไว้) แทนที่ EXCEPT ด้วย MINUS HTH.
Doug Clutter

20

dietbuddha มีคำตอบที่ดี ในกรณีที่คุณไม่มี MINUS หรือ EXCEPT ทางเลือกหนึ่งคือทำการรวมทั้งหมดระหว่างตารางจัดกลุ่มตามคอลัมน์ทั้งหมดและตรวจสอบให้แน่ใจว่ามีสองอย่าง:

SELECT col1, col2, col3
FROM
(SELECT * FROM tableA
UNION ALL  
SELECT * FROM tableB) data
GROUP BY col1, col2, col3
HAVING count(*)!=2

ฉันได้ลองใช้สิ่งนี้แล้ว (ได้มาจากบล็อกเซิร์ฟเวอร์ SQL ของ Jeff ) แต่ฉันต้องการแสดงรายการทั้งสองแถวจาก TableA และ TableB เพื่อให้ฉันสามารถมองเห็นความแตกต่างในแถวได้ คุณช่วยอธิบายวิธีทำได้ไหม
Emmanuel F

@ ตัวแทนนี่ฟังดูเป็นคำถามต่างหาก ฉันขอแนะนำให้ลงรายการเพื่อให้คนอื่นเห็นแทนที่จะแสดงความคิดเห็นที่นี่
jzd

เสร็จแล้ว และเสร็จแล้ว เปรียบเทียบค่าของ 2 ตารางและรายชื่อแถวที่จะแตกต่างกัน หวังว่าฉันจะได้รับผลลัพธ์ที่ยอดเยี่ยม :)
Emmanuel F

สิ่งนี้ทำงานได้ดีในสภาพแวดล้อม SQL ที่ จำกัด เช่น Visual FoxPro ขอบคุณ!
Kit Roed

1
เพียงแค่ตรวจทานนี้ ควรค่าแก่การกล่าวถึงว่าคีย์หลักช่วยให้แน่ใจว่ามีการบันทึกที่ไม่ซ้ำกันในตาราง ว่าถ้าตาราง (หรือคิวรี) อาจมีแถวที่ซ้ำกันDISTINCT/ GROUP BYแนะนำสำหรับเคียวรีย่อยในสหภาพเพื่อให้แน่ใจว่ามีเพียงหนึ่งระเบียนต่อตาราง มิฉะนั้น TableA อาจมี 2 ระเบียนและ TableB อาจมี 0 และไม่ตรงตามเงื่อนไข HAVING
vol7ron

8
SELECT c.ID
FROM clients c
WHERE EXISTS(SELECT c2.ID 
FROM clients2 c2
WHERE c2.ID = c.ID);

จะส่งคืน ID ทั้งหมดที่เป็น SAME ในทั้งสองตาราง หากต้องการรับความแตกต่างให้เปลี่ยน EXISTS เป็น NOT EXISTS


3

รับสคริปต์จากวันที่แล้วฉันแก้ไขมันเพื่อแสดงว่าแต่ละรายการมาจากตารางใด

DECLARE @table1 NVARCHAR(80)= 'table 1 name'
DECLARE @table2 NVARCHAR(80)= 'table 2 name'
DECLARE @sql NVARCHAR (1000)

SET @sql = 
'
SELECT ''' + @table1 + ''' AS table_name,* FROM
(
SELECT * FROM ' + @table1 + '
EXCEPT
SELECT * FROM ' + @table2 + '
) x

UNION 

SELECT ''' + @table2 + ''' AS table_name,* FROM 
(
SELECT * FROM ' + @table2 + '
EXCEPT
SELECT * FROM ' + @table1 + '
) y
'

EXEC sp_executesql @stmt = @sql

2

เพื่อให้เสร็จสมบูรณ์ proc จัดเก็บโดยใช้วิธีการยกเว้นเพื่อเปรียบเทียบ 2 ตารางและให้ผลลัพธ์ในตารางเดียวกันโดยมีสถานะข้อผิดพลาด 3 รายการ, ADD, DEL, GAP ตารางต้องมี PK เดียวกันคุณประกาศ 2 ตารางและฟิลด์เพื่อเปรียบเทียบ 1 หรือทั้งสองตาราง

เพียงใช้ ps_TableGap 'tbl1', 'Tbl2', 'fld1, fld2, fld3', 'fld4'fld5'fld6' (ไม่บังคับ)

/****** Object:  StoredProcedure [dbo].[ps_TableGap]    Script Date: 10/03/2013 16:03:44 ******/
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO

-- =============================================
-- Author:       Arnaud ALLAVENA
-- Create date: 03.10.2013
-- Description: Compare tables
-- =============================================
create PROCEDURE [dbo].[ps_TableGap]
    -- Add the parameters for the stored procedure here
    @Tbl1 as varchar(100),@Tbl2 as varchar(100),@Fld1 as varchar(1000), @Fld2 as varchar(1000)= ''
AS
BEGIN
    -- SET NOCOUNT ON added to prevent extra result sets from
    -- interfering with SELECT statements.

    SET NOCOUNT ON;
--Variables
--@Tbl1 = table 1
--@Tbl2 = table 2
--@Fld1 = Fields to compare from table 1
--@Fld2 Fields to compare from table 2
Declare @SQL varchar(8000)= '' --SQL statements
Declare @nLoop int = 1 --loop counter
Declare @Pk varchar(1000)= '' --primary key(s) 
Declare @Pk1 varchar(1000)= '' --first field of primary key
declare @strTmp varchar(50) = '' --returns value in Pk determination
declare @FldTmp varchar (1000) = '' --temporarily fields for alias calculation

--If @Fld2 empty we take @Fld1
--fields rules: fields to be compare must be in same order and type - always returns Gap
If @Fld2 = '' Set @Fld2 = @Fld1

--Change @Fld2 with Alias prefix xxx become _xxx 
while charindex(',',@Fld2)>0
begin
    Set @FldTmp = @FldTmp + (select substring(@Fld2,1,charindex(',',@Fld2)-1) + ' as _' + substring(@Fld2,1,charindex(',',@Fld2)-1) + ',')
    Set @Fld2 = (select ltrim(right(@Fld2,len(@Fld2)-charindex(',',@Fld2))))
end
Set @FldTmp = @FldTmp + @Fld2 + ' as _' + @Fld2
Set @Fld2 = @FldTmp

--Determinate primary key jointure
--rule: same pk in both tables
Set @nLoop = 1
Set @SQL = 'Declare crsr cursor for select COLUMN_NAME from INFORMATION_SCHEMA.KEY_COLUMN_USAGE where TABLE_NAME = '''
 + @Tbl1 + ''' or TABLE_SCHEMA + ''.'' + TABLE_NAME = ''' + @Tbl1 +  ''' or TABLE_CATALOG + ''.'' + TABLE_SCHEMA + ''.'' + TABLE_NAME = ''' + @Tbl1 
 + ''' order by ORDINAL_POSITION'
exec(@SQL)
open crsr 
fetch next from crsr into @strTmp
while @@fetch_status = 0
begin 
    if @nLoop = 1 
    begin 
        Set @Pk = 's.' + @strTmp + ' = b._' + @strTmp
        Set @Pk1 = @strTmp
        set @nLoop = @nLoop + 1 
    end 
    Else
    Set @Pk = @Pk + ' and s.' + @strTmp + ' = b._' + @strTmp
fetch next from crsr into @strTmp 

end 
close crsr
deallocate crsr

--SQL statement build
set @SQL = 'select case when s.' + @Pk1 + ' is null then ''Del'' when b._' + @Pk1 + ' is null then ''Add'' else ''Gap'' end as TypErr, '''
set @SQL = @SQL + @Tbl1 +''' as Tbl1, s.*, ''' + @Tbl2 +''' as Tbl2 ,b.* from (Select ' + @Fld1 + ' from ' + @Tbl1
set @SQL = @SQL + ' EXCEPT SELECT ' + @Fld2 + ' from ' + @Tbl2 + ')s full join (Select ' + @Fld2 + ' from ' + @Tbl2 
set @SQL = @SQL + ' EXCEPT SELECT ' + @Fld1 + ' from ' + @Tbl1 +')b on '+ @Pk 

--Run SQL statement
Exec(@SQL)
END

2

ที่มา: ใช้ NATURAL FULL JOIN เพื่อเปรียบเทียบสองตารางใน SQL โดย Lukas Eder

วิธีการที่ชาญฉลาดในการใช้NATURAL FULL JOINเพื่อตรวจจับแถวเดียวกัน / ต่างกันระหว่างสองตาราง

ตัวอย่างที่ 1 - ค่าสถานะสถานะ:

SELECT t1.*, t2.*, CASE WHEN t1 IS NULL OR t2 IS NULL THEN 'Not equal' ELSE 'Equal' END
FROM t1
NATURAL FULL JOIN t2;

ตัวอย่างที่ 2 - การกรองแถว

SELECT *
FROM (SELECT 't1' AS t1, t1.* FROM t1) t1 
NATURAL FULL JOIN (SELECT 't2' AS t2, t2.* FROM t2) t2 
WHERE t1 IS NULL OR t2 IS NULL -- show differences
--WHERE  t1 IS NOT NULL AND t2 IS NOT NULL    -- show the same

db <> การสาธิตซอ


1

คำตอบของพุทธบุตร ...

select * from
(
    select * from tableA
    minus
    select * from tableB
)
union all
select * from
(
    select * from tableB
    minus
    select * from tableA
)

1

คุณสามารถค้นหาความแตกต่างของ 2 ตารางโดยใช้การรวมกันของการแทรกทั้งหมดและการรวมภายนอกแบบเต็มใน Oracle ใน sql คุณสามารถแยกความแตกต่างผ่านการรวมภายนอกแบบเต็ม แต่ดูเหมือนว่า insert all / first ไม่มีอยู่ใน sql! ดังนั้นคุณต้องใช้แบบสอบถามต่อไปนี้แทน:

select * from A
full outer join B on
A.pk=B.pk
where A.field1!=B.field1
or A.field2!=B.field2 or A.field3!=B.field3 or A.field4!=B.field4 
--and A.Date==Date1

แม้ว่าการใช้ "OR" ในกรณีที่ไม่แนะนำให้ใช้ clause และโดยปกติจะให้ผลการดำเนินงานที่ต่ำกว่า แต่คุณยังสามารถใช้ข้อความค้นหาด้านบนได้หากตารางของคุณไม่ใหญ่มาก หากมีผลลัพธ์สำหรับข้อความค้นหาข้างต้นแสดงว่ามีความแตกต่างของ 2 ตารางตรงตามการเปรียบเทียบฟิลด์ 1,2,3,4 สำหรับการปรับปรุงประสิทธิภาพการสืบค้นคุณสามารถกรองตามวันที่ได้เช่นกัน (ตรวจสอบส่วนที่แสดงความคิดเห็น)


0
    SELECT unnest(ARRAY[1,2,2,3,3]) 
    EXCEPT
    SELECT unnest(ARRAY[1,1,2,3,3])
UNION
    SELECT unnest(ARRAY[1,1,2,3,3])
    EXCEPT
    SELECT unnest(ARRAY[1,2,2,3,3])

ผลลัพธ์เป็นโมฆะ แต่แหล่งที่มาแตกต่างกัน!

แต่:

(
    SELECT unnest(ARRAY[1,2,2,3])
    EXCEPT ALL
    SELECT unnest(ARRAY[2,1,2,3])
)
UNION
(
    SELECT unnest(ARRAY[2,1,2,3])
    EXCEPT ALL
    SELECT unnest(ARRAY[1,2,2,3])
)

ได้ผล


0

ฉันมีปัญหาเดียวกันนี้ใน SQL Server และเขียนสคริปต์ T-SQL นี้เพื่อทำให้กระบวนการเป็นไปโดยอัตโนมัติ (จริงๆแล้วนี่เป็นเวอร์ชันที่ลงน้ำฉันเขียนความแตกต่างทั้งหมดลงในตารางเดียวเพื่อให้รายงานได้ง่าย)

อัปเดต 'MyTable' และ 'MyOtherTable' เป็นชื่อของตารางที่คุณต้องการเปรียบเทียบ

DECLARE @ColName varchar(100)
DECLARE @Table1 varchar(100) = 'MyTable'
DECLARE @Table2 varchar(100) = 'MyOtherTable'


IF (OBJECT_ID('tempdb..#col') IS NOT NULL) DROP TABLE #col
SELECT  IDENTITY(INT, 1, 1) RowNum , c.name
INTO    #col
FROM    SYS.Objects o 
        JOIN SYS.columns c on o.object_id = c.object_id
WHERE   o.name = @Table1 AND NOT c.Name IN ('List','Columns','YouWantToIgnore')

DECLARE @Counter INT = (SELECT MAX(RowNum) FROM #col)

    WHILE @Counter > 0

        BEGIN
            SET @ColName = (SELECT name FROM #Col WHERE RowNum= @Counter)
                EXEC ('SELECT  t1.Identifier
                        ,t1.'+@ColName+' AS '+@Table1+@ColName+'
                        ,t2.'+@ColName+' AS '+@Table2+@ColName+'
                FROM    '+@Table1+' t1
                        LEFT JOIN '+@Table2+' t2 ON t1.Identifier = t2.Identifier 
                WHERE   t1.'+@ColName+' <> t2.'+@ColName)
            SET @Counter = @Counter - 1 
        END

0

ฉันเขียนสิ่งนี้เพื่อเปรียบเทียบผลลัพธ์ของมุมมองที่น่ารังเกียจที่ฉันย้ายจาก Oracle ไปยัง SQL Server มันสร้างตารางชั่วคราวคู่ #DataVariances และ #SchemaVariances โดยมีความแตกต่างใน (คุณเดาได้) ข้อมูลในตารางและสคีมาของตารางเอง

ต้องการให้ทั้งสองตารางมีคีย์หลัก แต่คุณสามารถวางลงใน tempdb พร้อมกับคอลัมน์ข้อมูลประจำตัวได้หากตารางต้นทางไม่มี

declare @TableA_ThreePartName nvarchar(max) = ''
declare @TableB_ThreePartName nvarchar(max) = ''
declare @KeyName nvarchar(max) = ''

/***********************************************************************************************

    Script to compare two tables and return differneces in schema and data.

    Author: Devin Lamothe       2017-08-11

***********************************************************************************************/
set nocount on

-- Split three part name into database/schema/table
declare @Database_A nvarchar(max) = (
    select  left(@TableA_ThreePartName,charindex('.',@TableA_ThreePartName) - 1))
declare @Table_A nvarchar(max) = (
    select  right(@TableA_ThreePartName,len(@TableA_ThreePartName) - charindex('.',@TableA_ThreePartName,len(@Database_A) + 2)))
declare @Schema_A nvarchar(max) = (
    select  replace(replace(@TableA_ThreePartName,@Database_A + '.',''),'.' + @Table_A,''))

declare @Database_B nvarchar(max) = (
    select  left(@TableB_ThreePartName,charindex('.',@TableB_ThreePartName) - 1))
declare @Table_B nvarchar(max) = (
    select  right(@TableB_ThreePartName,len(@TableB_ThreePartName) - charindex('.',@TableB_ThreePartName,len(@Database_B) + 2)))
declare @Schema_B nvarchar(max) = (
    select  replace(replace(@TableB_ThreePartName,@Database_B + '.',''),'.' + @Table_B,''))

-- Get schema for both tables
declare @GetTableADetails nvarchar(max) = '
    use [' + @Database_A +']
        select  COLUMN_NAME
             ,  DATA_TYPE
          from  INFORMATION_SCHEMA.COLUMNS
         where  TABLE_NAME = ''' + @Table_A + '''
           and  TABLE_SCHEMA = ''' + @Schema_A + '''
    '
create table #Table_A_Details (
    ColumnName nvarchar(max)
,   DataType nvarchar(max)
)
insert into #Table_A_Details
exec (@GetTableADetails)

declare @GetTableBDetails nvarchar(max) = '
    use [' + @Database_B +']
        select  COLUMN_NAME
             ,  DATA_TYPE
          from  INFORMATION_SCHEMA.COLUMNS
         where  TABLE_NAME = ''' + @Table_B + '''
           and  TABLE_SCHEMA = ''' + @Schema_B + '''
    '
create table #Table_B_Details (
    ColumnName nvarchar(max)
,   DataType nvarchar(max)
)
insert into #Table_B_Details
exec (@GetTableBDetails)


-- Get differences in table schema
            select  ROW_NUMBER() over (order by
                        a.ColumnName
                    ,   b.ColumnName) as RowKey
                 ,  a.ColumnName as A_ColumnName
                 ,  a.DataType as A_DataType
                 ,  b.ColumnName as B_ColumnName
                 ,  b.DataType as B_DataType
              into  #FieldList
              from  #Table_A_Details a
   full outer join  #Table_B_Details b
                on  a.ColumnName = b.ColumnName
             where  a.ColumnName is null
                or  b.ColumnName is null
                or  a.DataType <> b.DataType

        drop table  #Table_A_Details
        drop table  #Table_B_Details

            select  coalesce(A_ColumnName,B_ColumnName) as ColumnName
                 ,  A_DataType
                 ,  B_DataType
              into  #SchemaVariances
              from  #FieldList

-- Get differences in table data
declare @LastColumn int = (select max(RowKey) from #FieldList)
declare @RowNumber int = 1
declare @ThisField nvarchar(max)
declare @TestSql nvarchar(max)



create table #DataVariances (
    TableKey            nvarchar(max)
,   FieldName           nvarchar(max)
,   TableA_Value        nvarchar(max)
,   TableB_Value        nvarchar(max)
)

delete from #FieldList where A_DataType in ('varbinary','image') or B_DataType in ('varbinary','image') 

while @RowNumber <= @LastColumn begin
    set @TestSql = '
        select  coalesce(a.[' + @KeyName + '],b.[' + @KeyName + ']) as TableKey
             ,  ''' + @ThisField + ''' as FieldName
             ,  a.[' + @ThisField + '] as [TableA_Value]
             ,  b.[' + @ThisField + '] as [TableB_Value]
          from  [' + @Database_A + '].[' + @Schema_A + '].[' + @Table_A + '] a 
    inner join  [' + @Database_B + '].[' + @Schema_B + '].[' + @Table_B + '] b
            on  a.[' + @KeyName + '] = b.[' + @KeyName + ']
         where  ltrim(rtrim(a.[' + @ThisField + '])) <> ltrim(rtrim(b.[' + @ThisField + ']))
            or (a.[' + @ThisField + '] is null and  b.[' + @ThisField + '] is not null)
            or (a.[' + @ThisField + '] is not null and  b.[' + @ThisField + '] is null)
'

insert into #DataVariances
exec (@TestSql)

set @RowNumber = @RowNumber + 1
set @ThisField = (select coalesce(A_ColumnName,B_ColumnName) from #FieldList a where RowKey = @RowNumber)

end

drop table #FieldList

print 'Query complete.  Select from #DataVariances to verify data integrity or #SchemaVariances to verify schemas match.  Data types varbinary and image are not checked.'

0

คำตอบส่วนใหญ่ดูเหมือนจะเพิกเฉยต่อปัญหาที่คามิลยกขึ้นมา (นั่นคือที่ที่ตารางมีแถวที่เหมือนกัน แต่มีการซ้ำกันในแต่ละตาราง) น่าเสียดายที่ฉันไม่สามารถใช้โซลูชันของเขาได้เนื่องจากฉันอยู่ใน Oracle สิ่งที่ดีที่สุดที่ฉันสามารถทำได้คือ:

SELECT * FROM
   (
   SELECT column1, column2, ..., COUNT(*) AS the_count
   FROM tableA
   GROUP BY column1, column2, ...
   MINUS
   SELECT column1, column2, ..., COUNT(*) AS the_count
   FROM tableB
   GROUP BY column1, column2, ...
   )
UNION ALL
   (
   SELECT column1, column2, ..., COUNT(*) AS the_count
   FROM tableB
   GROUP BY column1, column2, ...
   MINUS
   SELECT column1, column2, ..., COUNT(*) AS the_count
   FROM tableA
   GROUP BY column1, column2, ...
   )

0

เพื่อเปรียบเทียบ T1 (PK, A, B) และ T2 (PK, A, B)

ขั้นแรกให้เปรียบเทียบชุดคีย์หลักเพื่อค้นหาค่าคีย์ที่หายไปในด้านใดด้านหนึ่ง:

SELECT T1.*, T2.* FROM T1 FULL OUTER JOIN T2 ON T1.PK=T2.PK WHERE T1.PK IS NULL OR T2.PK IS NULL;

จากนั้นแสดงรายการค่าทั้งหมดไม่ตรงกัน:

SELECT T1.PK, 'A' AS columnName, T1.A AS leftValue, T2.A AS rightValue FROM T1 JOIN T2 ON T1.PK=T2.PK WHERE COALESCE(T1.A,0) != COALESCE(T2.A,0)
UNION ALL
SELECT T1.PK, 'B' AS columnName, T1.B AS leftValue, T2.B AS rightValue FROM T1 JOIN T2 ON T1.PK=T2.PK WHERE COALESCE(T1.B,0) != COALESCE(T2.B,0)

A และ B ต้องเป็นประเภทเดียวกัน คุณสามารถใช้แผนข้อมูลเพื่อสร้างการเลือก อย่าลืม COALESCE รวมผลลัพธ์ IS NULL ไว้ด้วย คุณยังสามารถใช้ FULL OUTER JOIN และ COALESCE (T1.PK, 0) = COALESCE (T2.PK, 0)

ตัวอย่างสำหรับคอลัมน์ประเภท varchar:

SELECT concat('SELECT T1.PK, ''', COLUMN_NAME, ''' AS columnName, T1.', COLUMN_NAME, ' AS leftValue, T2.', COLUMN_NAME, ' AS rightValue FROM T1 JOIN T2 ON T1.PK=T2.PK WHERE COALESCE(T1.',COLUMN_NAME, ',0)!=COALESCE(T2.', COLUMN_NAME, ',0)')
FROM INFORMATION_SCHEMA.COLUMNS 
WHERE TABLE_NAME='T1' AND DATA_TYPE IN ('nvarchar','varchar');

0

เราสามารถเปรียบเทียบข้อมูลจากตาราง DB2 สองตารางโดยใช้แบบสอบถามง่ายๆด้านล่าง

ขั้นตอนที่ 1: - เลือกคอลัมน์ทั้งหมดที่เราต้องการเปรียบเทียบจากตาราง (T1) ของสคีมา (S)

     SELECT T1.col1,T1.col3,T1.col5 from S.T1

ขั้นตอนที่ 2: - ใช้คำหลัก 'ลบ' เพื่อเปรียบเทียบ 2 ตาราง

ขั้นตอนที่ 3: - เลือกคอลัมน์ทั้งหมดที่เราต้องการเปรียบเทียบจากตาราง (T2) ของสคีมา (S)

     SELECT T2.col1,T2.col3,T2.col5 from S.T1

ผลลัพธ์ END:

     SELECT T1.col1,T1.col3,T1.col5 from S.T1
     MINUS 
     SELECT T2.col1,T2.col3,T2.col5 from S.T1;

หากคิวรีไม่ส่งคืนแถวข้อมูลจะเหมือนกันทุกประการ


-1

ใน MySQL ซึ่งไม่รองรับ "ลบ" และคำนึงถึงประสิทธิภาพการทำงานนี่เป็นวิธีที่รวดเร็ว

query:
SELECT 
t1.id, 
t1.id 
FROM t1 inner join t2 using (id) where concat(t1.C, t1.D, ...)<>concat(t2.C, t2.D, ...)

-1

อีกทางเลือกหนึ่งที่ได้รับการปรับปรุงตามคำตอบของ dietbuddha & IanMc คำค้นหามีคำอธิบายเพื่อช่วยในการแสดงตำแหน่งที่มีและขาดหายไป (NB: สำหรับSQL Server )

(
    select 'InTableA_NoMatchInTableB' as Msg, * from tableA
    except
    select 'InTableA_NoMatchInTableB' , * from tableB
)
union all
(
    select 'InTableB_NoMatchInTableA' as Msg, * from tableB
    except
    select 'InTableB_NNoMatchInTableA' ,* from tableA
)

-1
SELECT * 
FROM TABLE A
WHERE NOT EXISTS (SELECT 'X' 
                  FROM  TABLE B 
                  WHERE B.KEYFIELD1 = A.KEYFIELD1 
                  AND   B.KEYFIELD2 = A.KEYFIELD2 
                  AND   B.KEYFIELD3 = A.KEYFIELD3)
;

'X' คือค่าใด ๆ

สลับตารางเพื่อดูความคลาดเคลื่อนต่างๆ

อย่าลืมเข้าร่วมช่องสำคัญในตารางของคุณ

หรือเพียงแค่ใช้ตัวดำเนินการ MINUS กับ 2 คำสั่งที่เลือกอย่างไรก็ตาม MINUS สามารถทำงานได้ใน Oracle เท่านั้น


ไม่รองรับการใช้งานทั้งหมด (เช่นเซิร์ฟเวอร์เซิร์ฟเวอร์ใช้ยกเว้น)
Sir Swears-a-lot
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.