집합 연산자 사용

집합 연산자는 두 개 이상의 SELECT 문의 결과를 결합하는 데 사용됩니다. Oracle 11g에서 사용할 수있는 SET 연산자는 UNION, UNION ALL, INTERSECT 및 MINUS입니다.

UNION 집합 연산자는 두 SELECT 문의 결합 된 결과를 반환합니다. 본질적으로 결과에서 중복을 제거합니다. 즉, 각 중복 결과에 대해 하나의 행만 나열됩니다.이 동작에 대응하려면 중복을 유지하는 UNION ALL 집합 연산자를 사용합니다. 최종 결과 .INTERSECT는 두 SELECT 쿼리에 공통적 인 레코드 만 나열합니다. MINUS 집합 연산자는 두 번째 쿼리의 결과가 첫 번째 쿼리의 결과에서도 발견되는 경우 출력에서 ​​제거합니다. INTERSECT 및 MINUS 집합 연산은 중복되지 않은 결과를 생성합니다.

모든 SET 연산자는 동일한 우선 순위를 공유합니다. 대신 쿼리 실행 중에 Oracle은 왼쪽에서 오른쪽으로 또는 위에서 아래로 평가를 시작합니다. 명시 적으로 괄호를 사용하면 괄호가 우선 순위가 부여되므로 순서가 다를 수 있습니다. 매달린 연산자.

기억해야 할 사항-

  • 참여하는 모든 SELECT 문에서 동일한 수의 열을 선택해야합니다. 디스플레이에 사용 된 열 이름은 첫 번째 쿼리에서 가져옵니다.

  • 열 목록의 데이터 유형은 Oracle에서 호환 가능 / 암시 적으로 변환 가능해야합니다. Oracle은 구성 요소 쿼리의 해당 열이 다른 데이터 유형 그룹에 속하는 경우 암시 적 유형 변환을 수행하지 않습니다. 예를 들어 첫 번째 구성 요소 쿼리의 열이 데이터 유형 DATE이고 두 번째 구성 요소 쿼리의 해당 열이 데이터 인 경우 유형 CHAR, Oracle은 암시 적 변환을 수행하지 않지만 ORA-01790 오류가 발생합니다.

  • 결과 집합을 정렬하려면 위치 순서를 사용해야합니다. 집합 연산자에서는 개별 결과 집합 순서를 지정할 수 없습니다. ORDER BY는 쿼리 끝에 한 번 나타날 수 있습니다. 예를 들면

  • UNION 및 INTERSECT 연산자는 교환 적입니다. 즉, 쿼리 순서는 중요하지 않습니다. 최종 결과를 변경하지 않습니다.

  • 성능면에서 UNION ALL은 중복 필터링 및 결과 집합 정렬에 리소스가 낭비되지 않기 때문에 UNION에 비해 더 나은 성능을 보여줍니다.

  • 집합 연산자는 하위 쿼리의 일부가 될 수 있습니다.

  • 집합 연산자는 TABLE 컬렉션식이 포함 된 SELECT 문에서 사용할 수 없습니다.

  • LONG, BLOB, CLOB, BFILE, VARRAY 또는 중첩 테이블은 Set 연산자에서 사용할 수 없습니다 .For update 절은 집합 연산자와 함께 사용할 수 없습니다.

노동 조합

UNION 연산자를 사용하여 여러 SELECT 쿼리를 조인하면 Oracle은 NULL 값을 무시하지 않고 모든 중복을 제거한 후 정렬 된 순서 (기본적으로 오름차순)로 모든 복합 SELECT 쿼리의 결합 된 결과를 표시합니다.

UNION 연산자를 사용하여 조인 된 아래 5 개의 쿼리를 고려하십시오. 최종 결합 된 결과 집합에는 모든 SQL의 값이 포함됩니다. 중복 제거 및 데이터 정렬에 유의하십시오.

SELECT 1 NUM FROM DUAL
UNION
SELECT 5 FROM DUAL 
UNION
SELECT 3 FROM DUAL
UNION
SELECT 6 FROM DUAL
UNION
SELECT 3 FROM DUAL;

NUM
-------
1
3
5
6

참고로 SELECT 쿼리에서 선택한 열은 호환 가능한 데이터 유형이어야합니다. Oracle은 규칙을 위반하면 오류 메시지를 표시합니다.

SELECT TO_DATE('12-OCT-03') FROM DUAL
UNION
SELECT '13-OCT-03' FROM DUAL;

SELECT TO_DATE('12-OCT-03') FROM DUAL
       *
ERROR at line 1:
ORA-01790: expression must have same datatype as corresponding expression

UNION ALL

UNION과 UNION ALL은 기능면에서 약간의 차이를 제외하고 비슷합니다. 그러나 UNION ALL은 중복을 제거하고 데이터를 정렬하지 않고 결과 집합을 제공합니다. 예를 들어 위의 쿼리에서 UNION은 UNION ALL로 대체되어 효과를 확인합니다.

UNION 섹션에 설명 된 쿼리를 고려하십시오. 정렬 및 중복 제거없이 생성되는 출력의 차이에 유의하십시오.

SELECT 1 NUM FROM DUAL
UNION ALL
SELECT 5 FROM DUAL 
UNION ALL
SELECT 3 FROM DUAL
UNION ALL
SELECT 6 FROM DUAL
UNION ALL
SELECT 3 FROM DUAL;

NUM
-------
1
5
3
6
3

교차

INTERSECT 연산자를 사용하여 Oracle은 정렬 된 순서 (기본적으로 오름차순)로 정렬 된 중복 및 데이터없이 SELECT 문 모두의 공통 행을 표시합니다.

예를 들어, 아래 SELECT 쿼리는 부서 10과 20에서 공통되는 급여를 검색합니다. ISO SQL 표준에 따라 INTERSECT는 집합 연산자의 평가보다 우선 순위가 높지만 Oracle에서 여전히 통합하지는 않습니다.

SELECT SALARY
FROM employees
WHERE DEPARTMENT_ID = 10
INTRESECT
SELECT SALARY 
FROM employees
WHERE DEPARTMENT_ID = 20

SALARY
---------
1500
1200
2000

마이너스

빼기 연산자는 첫 번째 쿼리에는 있지만 두 번째 쿼리에는없는 행을 표시하며 중복 및 데이터는 기본적으로 오름차순으로 정렬되지 않습니다.

SELECT JOB_ID
FROM employees
WHERE DEPARTMENT_ID = 10
MINUS
SELECT JOB_ID
FROM employees
WHERE DEPARTMENT_ID = 20;

JOB_ID
-------------        
HR
FIN
ADMIN

SELECT 문 일치

복합 SELECT 문이 선택한 열의 개수와 데이터 유형이 다를 수있는 시나리오가있을 수 있습니다. 따라서 열 목록을 명시 적으로 일치시키기 위해 각 SELECT 문에서 선택한 열의 개수 및 데이터 유형과 일치하도록 누락 된 위치에 NULL 열이 삽입됩니다. 숫자 열의 경우 쿼리에서 선택한 열 유형과 일치하도록 0을 대체 할 수도 있습니다.

아래 쿼리에서 직원 이름 (varchar2)과 위치 ID (번호)의 데이터 유형이 일치하지 않습니다. 따라서 아래 쿼리를 실행하면 호환성 문제로 인해 오류가 발생합니다.

SELECT DEPARTMENT_ID "Dept", first_name "Employee"
FROM employees
UNION
SELECT DEPARTMENT_ID, LOCATION_ID
FROM departments;

ERROR at line 1:
ORA-01790: expression must have same datatype as corresponding expression

명시 적으로 위치 ID와 직원 이름을 NULL로 대체하여 열을 일치시킬 수 있습니다.

SELECT DEPARTMENT_ID "Dept", first_name "Employee", NULL "Location"
FROM employees
UNION
SELECT DEPARTMENT_ID, NULL "Employee", LOCATION_ID
FROM departments;

SET 작업에서 ORDER BY 절 사용

ORDER BY 절은 복합 SELECT 문을 포함하는 쿼리 끝에 한 번만 나타날 수 있으며 개별 SELECT 문에는 ORDER BY 절이있을 수 없음을 의미합니다. 또한 정렬은 첫 번째 SELECT 쿼리에만 나타나는 열을 기반으로 할 수 있습니다. 따라서 열 위치를 사용하여 복합 쿼리를 정렬하는 것이 좋습니다.

아래의 compund 쿼리는 두 부서의 결과를 통합하고 SALARY 열을 기준으로 정렬합니다.

SELECT employee_id, first_name, salary
FROM employees
WHERE department_id=10
UNION
SELECT employee_id, first_name, salary
FROM employees
WHERE department_id=20
ORDER BY 3;