123. DML - JOIN
123.1 JOIN
2개의 릴레이션에서 연관된 튜플들을 결합하여, 하나의 새로운 릴레이션을 반환
- JOIN은 일반적으로 FROM절에 기술하지만, 릴레이션이 사용되는 곳 어디에나 사용할 수 있음
- JOIN은 크게 INNER JOIN과 OUTER JOIN으로 구분
123.2 INNER JOIN
일반적으로 EQUI JOIN과 NON-EQUI JOIN으로 구분
- 조건이 없는 INNER JOIN을 수행하면 CROSS JOIN과 동일한 결과를 얻을 수 있음
- CROSS JOIN(교차 조인) : 교차 조인은 조인하는 두 테이블에 있는 튜플들의 순서쌍을 결과로 반환
- 교차 조인의 결과로 반환되는 테이블의 행의 수는 두 테이블의 행의 수를 곱한 것과 같음
123.2.1 EQUI JOIN
JOIN 대상 테이블에서 공통 속성을 기준으로 '='(equal) 비교에 의해 같은 값을 가지는 행을 연결하여 결과를 생성하는 JOIN 방법
- EQUI JOIN에서 JOIN 조건이 '='일 때 동일한 속성이 두 번 나타나게 되는데, 이 중 중복된 속성을 제거하여 같은 속성을 한 번만 표기하는 방법을 NATURAL JOIN이라고 함
- EQUI JOIN에서 연결 고리가 되는 공통 속성을 JOIN 속성이라고 함
- WHERE절을 이용한 EQUI JOIN의 표기 형식
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1, 테이블명2, ···
WHERE 테이블명1.속성명 = 테이블명2.속성명;
- NATURAL JOIN절을 이용한 EQUI JOIN의 표기 형식
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1 NATURAL JOIN 테이블명2;
- JOIN ~ USING절을 이용한 EQUI JOIN의 표기 형식
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1 JOIN 테이블명2 USING(속성명);
학번 |
이름 |
학과코드 |
선배 |
성적 |
15 |
고길동 |
com |
|
83 |
16 |
이순신 |
han |
|
96 |
17 |
김선달 |
com |
15 |
95 |
19 |
아무개 |
han |
16 |
75 |
37 |
박치민 |
|
17 |
55 |
학과코드 |
학과명 |
com |
컴퓨터 |
han |
국어 |
eng |
영어 |
등급 |
최저 |
최고 |
A |
90 |
100 |
B |
80 |
89 |
C |
60 |
79 |
D |
0 |
59 |
123.2.1.1 예제 1
- <학생> 테이블과 <학과> 테이블에서 '학과코드'값이 같은 튜플을 JOIN하여 '학번', '이름', '학과코드', '학과명'을 출력하는 SQL문을 작성하라
SELECT 학번, 이름, 학생.학과코드, 학과명 FROM 학생, 학과 WHERE 학생.학과코드=학과.학과코드;
SELECT 학번, 이름, 학생.학과코드, 학과명 FROM 학생 NATURAL JOIN 학과;
SELECT 학번, 이름, 학생.학과코드, 학과명 FROM 학생 JOIN 학과 USING(학과코드);
학번 |
이름 |
학과코드 |
학과명 |
15 |
고길동 |
com |
컴퓨터 |
16 |
이순신 |
han |
국어 |
17 |
김선달 |
com |
컴퓨터 |
19 |
아무개 |
han |
국어 |
123.2.2 NON-EQUI JOIN
JOIN 조건에 '='조건이 아닌 나머지 비교 연산자, 즉 >, <, <>, >=, <= 연산자를 사용하는 JOIN 방법
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1, 테이블명2, ···
WHERE (NON-EQUI JOIN 조건);
123.2.2.1 예제 2
- <학생> 테이블과 <성적등급> 테이블을 JOIN하여 각 학생의 '학번', '이름', '성적', '등급'을 출력하는 SQL문을 작성
SELECT 학번, 이름, 성적, 등급
FROM 학생, 성적등급
WHERE 학생.성적 BETWEEN 성적등급.최저 AND 성적등급.최고;
학번 |
이름 |
성적 |
등급 |
15 |
고길동 |
83 |
B |
16 |
이순신 |
96 |
A |
17 |
김선달 |
95 |
A |
19 |
아무개 |
75 |
C |
37 |
박치민 |
55 |
D |
123.3 OUTER JOIN
릴레이션에서 JOIN 조건에 만족하지 않는 튜플도 결과로 출력하기 위한 JOIN 방법으로, LEFT OUTER JOIN, RIGHT OUTER JOIN, FULL OUTER JOIN이 존재
123.3.1 LEFT OUTER JOIN
INNER JOIN의 결과를 구한 후, 우측 항 릴레이션의 어떤 튜플과도 맞지 않는 좌측 항의 릴레이션에 있는 튜플들에 NULL값을 붙여서 INNER JOIN의 결과에 추가
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1 LEFT OUTER JOIN 테이블명2
ON 테이블명1.속성명=테이블명2.속성명;
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1, 테이블명2
WHERE 테이블명1.속성명=테이블명2.속성명(+);
123.3.2 RIGHT OUTER JOIN
INNER JOIN의 결과를 구한 후, 좌측 항 릴레이션의 어떤 튜플과도 맞지 않는 우측 항의 릴레이션에 있는 튜플들에 NULL값을 붙여서 INNER JOIN의 결과에 추가
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1 RIGHT OUTER JOIN 테이블명2
ON 테이블명1.속성명=테이블명2.속성명;
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1, 테이블명2
WHERE 테이블명1.속성명(+)=테이블명2.속성명;
123.3.3 FULL OUTER JOIN
LEFT OUTER JOIN과 RIGHT OUTER JOIN을 합쳐 놓은 것
- INNER JOIN의 결과를 구한 후, 좌측 항의 릴레이션의 튜플들에 대해 우측 항의 릴레이션의 어떤 튜플과도 맞지 않는 튜플들에 NULL값을 붙여서 INNER JOIN의 결과에 추가
- 그리고 유사하게 우측 항의 릴레이션의 튜플들에 대해 좌측 항의 릴레이션의 어떤 튜플과도 맞지 않는 튜플들에 NULL값을 붙여서 INNER JOIN의 결과에 추가
- 표기 형식
SELECT [테이블명1.]속성명, [테이블명2.]속성명, ···
FROM 테이블명1 FULL OUTER JOIN 테이블명2
ON 테이블명1.속성명=테이블명2.속성명;
123.3.4 예제 1
- <학생> 테이블과 <학과> 테이블에서 '학과코드' 값이 같은 튜플을 JOIN하여 '학번', '이름', '학과코드', '학과명'을 출력하는 SQL문을 작성하라
- 이때, '학과코드'가 입력되지 않은 학생도 출력하라
SELECT 학번, 이름, 학생.학과코드, 학과명
FROM 학생 LEFT OUTER JOIN 학과
ON 학생.학과코드=학과.학과코드;
SELECT 학번, 이름, 학생.학과코드, 학과명
FROM 학생, 학과
WHERE 학생.학과코드=학과.학과코드(+);
SELECT 학번, 이름, 학생.학과코드, 학과명
FROM 학생 RIGHT OUTER JOIN 학과
ON 학생.학과코드=학과.학과코드;
SELECT 학번, 이름, 학생.학과코드, 학과명
FROM 학생, 학과
WHERE 학생.학과코드(+)=학과.학과코드;
학번 |
이름 |
학과코드 |
학과명 |
15 |
고길동 |
com |
컴퓨터 |
16 |
이순신 |
han |
국어 |
17 |
김선달 |
com |
컴퓨터 |
19 |
아무개 |
han |
국어 |
37 |
박치민 |
|
|
123.3.5 예제 2
- <학생> 테이블과 <학과> 테이블에서 '학과코드' 값이 같은 튜플을 JOIN하여 '학번', '이름', '학과코드', '학과명'을 출력하는 SQL문을 작성하라
- 이때, '학과코드'가 입력 안 된 학생이나 학생이 없는 '학과코드'도 모두 출력하라
SELECT 학번, 이름, 학과.학과코드, 학과명
FROM 학생 FULL OUTER JOIN 학과
ON 학생.학과코드=학과.학과코드;
학번 |
이름 |
학과코드 |
학과명 |
15 |
고길동 |
com |
컴퓨터 |
16 |
이순신 |
han |
국어 |
17 |
김선달 |
com |
컴퓨터 |
19 |
아무개 |
han |
국어 |
37 |
박치민 |
|
|
|
|
eng |
영어 |