セット演算子 - EXCEPT および INTERSECT (Transact-SQL)

Applies to: SQL Server Azure SQL Database Azure SQL Managed Instance Azure Synapse Analytics Analytics Platform System (PDW) Microsoft Fabric の SQL 分析エンドポイント Microsoft Fabric のウェアハウス

2 つのクエリの結果を比較して、個別の行を返します。

EXCEPT は、左の入力クエリからの行のうち、右の入力クエリから出力されないものを、重複を除去した上で返します。

INTERSECT では、両方の左辺と右辺の入力クエリ演算子によって出力される個別の行を返します。

EXCEPT と INTERSECT を使用する 2 つのクエリの結果セットを結合する場合の基本的な規則は、次のとおりです。

  • 列の数と順番は、すべてのクエリで同じであること

  • データ型に互換性があること

Transact-SQL 構文表記規則

構文

{ <query_specification> | ( <query_expression> ) }   
{ EXCEPT | INTERSECT }  
{ <query_specification> | ( <query_expression> ) }  

引数

<query_specification> | ( <query_expression> )
データを返すクエリ定義またはクエリ式を指定します。このデータが、別のクエリ定義またはクエリ式で返されるデータと比較されます。 EXCEPT または INTERSECT 演算の一部である列の定義は同じである必要はありません。 ただし、暗黙的な変換によって比較できる必要があります。 データ型が異なるとき、比較するために実行されるデータの型は、データ型の優先順位ルールに基づいて決定されます。

型は同じだが、有効桁数、小数点以下桁数、または長さが異なる場合、結果は式の結合と同じルールに基づいて決定されます。 詳しくは、「有効桁数、小数点以下桁数、および長さ (Transact-SQL)」をご覧ください。

xmltextntextimage、またはバイナリ以外の CLR ユーザー定義型の列は比較できないため、クエリ定義またはクエリ式ではそれらを返すことはできません。

EXCEPT
EXCEPT 演算子の左のクエリから個別の値を返します。 右のクエリでは返されない値が返されます。

INTERSECT
INTERSECT 演算子の左右両方のクエリによって返される個別の値を返します。

解説

比較可能な列のデータ型は、EXCEPT または INTERSECT 演算の左右のクエリによって返されます。 これらのデータ型には、照合順序が異なる文字データ型を含めることができます。 これを行うと、必要な比較は、照合順序の優先順位ルールに従って実行されます。 この変換を実行できない場合、SQL Server データベース エンジン によってエラーが返されます。

DISTINCT 行を決定するために列の値を比較するとき、2 つの NULL 値は等しいと見なされます。

EXCEPT と INTERSECT では、演算子の左のクエリで返される結果セットの列名と同じ列名が返されます。

ORDER BY 句の列名または列の別名は、左のクエリで返される列名を参照している必要があります。

EXCEPT または INTERSECT で返される結果セットの列で NULL 値が許容されるかどうかは、演算子の左のクエリで返される対応する列の設定と同じになります。

EXCEPT または INTERSECT を他の演算子と共に 1 つの式で使用する場合、評価は次の優先順で行われます。

  1. かっこで囲んだ式

  2. INTERSECT 演算子

  3. EXCEPT と UNION (式の中の位置に基づいて、左から右の順に評価)

EXCEPT または INTERSECT を使用して、3 つ以上のクエリのセットを比較できます。 これを行う場合、データ型の変換は、1 度に 2 つのクエリを比較し、前に説明した式の評価のルールに従って決定されます。

EXCEPT と INTERSECT は、分散パーティション ビュー定義やクエリ通知では使用できません。

EXCEPT と INTERSECT は分散クエリで使用できますが、この場合、ローカル サーバーでのみ実行され、リンク サーバーにはプッシュされません。 そのため、EXCEPT と INTERSECT を分散クエリで使用すると、パフォーマンスに影響が生じる可能性があります。

高速順方向専用カーソルと静的カーソルを EXCEPT または INTERSECT 演算と共に使用した場合、結果セットでそれらを使用できます。 キーセット ドリブン カーソルまたは動的カーソルも、EXCEPT または INTERSECT 演算と一緒に使用できます。 これを行うと、演算の結果セットのカーソルが静的カーソルに変換されます。

EXCEPT 演算が、SQL Server Management Studio のグラフィカル プラン表示機能を使用して表示される場合、この演算は left anti semi join として表示され、INTERSECT 演算は left semi join として表示されます。

次の例は、INTERSECT 演算子と EXCEPT 演算子の使用方法を示しています。 最初のクエリでは、INTERSECT および EXCEPT の結果と比較するため、Production.Product テーブルからすべての値を返します。

-- Uses AdventureWorks  
  
SELECT ProductID   
FROM Production.Product ;  
--Result: 504 Rows  

次のクエリでは、INTERSECT 演算子の左右両方のクエリによって返される個別の値が返されます。

-- Uses AdventureWorks  
  
SELECT ProductID   
FROM Production.Product  
INTERSECT  
SELECT ProductID   
FROM Production.WorkOrder ;  
--Result: 238 Rows (products that have work orders)  

次のクエリでは、EXCEPT 演算子の左のクエリにあって、右のクエリにない個別の値が返されます。

-- Uses AdventureWorks  
  
SELECT ProductID   
FROM Production.Product  
EXCEPT  
SELECT ProductID   
FROM Production.WorkOrder ;  
--Result: 266 Rows (products without work orders)  

次のクエリでは、EXCEPT 演算子の左のクエリにあって、右のクエリにない個別の値が返されます。 テーブルは、前の例と逆になります。

-- Uses AdventureWorks  
  
SELECT ProductID   
FROM Production.WorkOrder  
EXCEPT  
SELECT ProductID   
FROM Production.Product ;  
--Result: 0 Rows (work orders without products)  

例: Azure Synapse Analytics、Analytics Platform System (PDW)

次の例は、INTERSECT 演算子と EXCEPT 演算子の使用方法を示しています。 最初のクエリでは、INTERSECT および EXCEPT の結果と比較するため、FactInternetSales テーブルからすべての値を返します。

-- Uses AdventureWorks  
  
SELECT CustomerKey   
FROM FactInternetSales;  
--Result: 60398 Rows  

次のクエリでは、INTERSECT 演算子の左右両方のクエリによって返される個別の値が返されます。

-- Uses AdventureWorks  
  
SELECT CustomerKey   
FROM FactInternetSales    
INTERSECT   
SELECT CustomerKey   
FROM DimCustomer   
WHERE DimCustomer.Gender = 'F'  
ORDER BY CustomerKey;  
--Result: 9133 Rows (Sales to customers that are female.)  

次のクエリでは、EXCEPT 演算子の左のクエリにあって、右のクエリにない個別の値が返されます。

-- Uses AdventureWorks  
  
SELECT CustomerKey   
FROM FactInternetSales    
EXCEPT   
SELECT CustomerKey   
FROM DimCustomer   
WHERE DimCustomer.Gender = 'F'  
ORDER BY CustomerKey;  
--Result: 9351 Rows (Sales to customers that are not female.)