본문 바로가기
DATABASE/ORACLE

[Oracle] SQL 서브쿼리

서브쿼리





하나의 테이블에서 검색한 결과를 다른 테이블에 전달하여 새로운 결과를 검색하는 경우 사용하는 서브쿼리

SQL 문장의 하부 절에 쿼리문을 사용 즉, 메인쿼리의 SELECT문 하부의 WHERE절이나 FROM절에서 사용되는 쿼리를 서브쿼리라고 한다


주의사항 연산자의 오른쪽에 위치 해야하며 괄호로 묶어주어야 한다




서브쿼리의 위치에 따른 명칭

● SELECT 문에 있는 서브쿼리 : 스칼라 서브쿼리

● FROM절에 있는 서브쿼리 : 인라인 뷰

● WHERE절에 있는 서브쿼리 : 서브쿼리

메인쿼리 서브쿼리

수정 : SELECT DEPNO => DEPTNO



1
SELECT DNAME FROM DEPT WHERE DEPTNO = 20;
cs



1
2
SELECT DNAME FROM DEPT WHERE DEPTNO = (SELECT DEPTNO FROM EMP WHERE ENAME='SCOTT');
 
cs


EMP 테이블에는 부서번호만 있지, 부서명은 없다, 부서명은 DEPT에 있다.




● 서브쿼리는 하나의 SELECT 문장의 절 안에 포함 된 또 하나의 SELECT 문장이다.

그렇기에 서브쿼리를 포함하고 있는 쿼리문을 메인 쿼리, 포함 된 또 하나의 쿼리를 서브쿼리라 한다.


● 서브쿼리는 비교 연산자의 오른쪽에 기술 해야 하고 반드시 괄호로 둘러 쌓아야 한다.

● 서브쿼리는 메인쿼리가 실행되기 이전에 한 번만 실행이 된다.

서브쿼리문을 먼저 실행하고, 메인쿼리가 실행하게 된다.





단일행 서브쿼리


 하나의 로우만을 반환되는 서브 쿼리의 결과를 메인 쿼리에 보내지게 되는데 메인 쿼리의 WHERE 절에서는 단일행 비교 연산자인 =,>,>=,<.<=,<>를 사용한다.

즉, 서브쿼리의 결과 1개의 값만 나오고 이 값을 메인쿼리로 전달 하는 것


● 단일행 서브쿼리에서는 단일행 비교 연산자인 =, <>, <, >등의 연산자를 사용 할 수 있다



예제) 

● DALLAS에서 근무하는 사원의 이름, 부서 번호를 출력 해보자.

사원의 이름과 부서 번호는 EMP 테이블에서 가지고 올 수 있으며 근무지 DALLAS는 DEPT 테이블에 있다.


1
SELECT ename, deptno  from emp where deptno = (select deptno from dept where loc='DALLAS');
cs



● SALES(영업) 부서에 근무하는 모든 사원의 이름과 급여를 출력 해보자.

마찬가지로 사원의 이름과 급여는 EMP 테이블에서 가지고 올 수 있으며 영업부는 DEPT 테이블에서 확인 할 수 있다.


1
select ename, sal from emp where deptno = (select deptno from dept where dname='SALES')

cs



● 직속 상관이 KING인 사원의 이름과 급여를 출력 해보자

1
select ename , sal from emp where mgr = (select empno from emp where ename = 'KING');
cs




다중행 서브쿼리

서브쿼리에서 반환 되는 결과가 하나 이상의 행일 때 사용하는 서브쿼리

결과가 2개 이상 구해지는 쿼리문을 서브쿼리로 기술 할 경우에는 다중행 연산자와 함께 사용


다중행 서브쿼리에서 사용 할 수 있는 다중행 비교 연산자

● IN : 같은 값을 찾음

● >ANY : 최소값을 반환

● <ALL : 최소값을 반환

● <ANY : 최대값을 반환

● >ALL : 최대값을 반환

● EXIST : 서브쿼리 값이 있을 경우 반환


IN 연산자는 단일행 비교 연산자 " = "과 비슷한 용도로 사용한다.

서브쿼리의 반환이 1행 이상일 때 그 행들의 값을 넘겨준다.



----------------------------------------------------------보류---

● 10번 부서에 근무하는 사원의 이름과 10번 부서의 부서명을 출력 해보자


다중행 서브쿼리, ( )의 서브 쿼리 결과를 d라고 부른다.

그 후 JOIN이 들어간다

1
2
3
select e.ename , d.dname from emp e,
(select deptno, dname from dept where deptno = 10) d
where e.deptno = d.deptno;
cs


JOIN만을 통해서 구할 수도 있다.

1
2
select e.ename, d.dname from emp e, dept d
where e.deptno = d.deptno and d.deptno = 10;
cs


-------------------------------------------------------------------



예제)

● 부서별로 가장 급여를 많이 받는 사원의정보(사원 번호, 사원 이름, 급여, 부서번호)를 출력 해보자


1
2
3
4
5
6
7
8
9
SQL> SELECT deptno, ename, sal FROM emp
  2  WHERE sal IN (SELECT max(sal) FROM emp group by deptno);
 
    DEPTNO ENAME             SAL
---------- ---------- ----------
        30 BLAKE            2850
        20 SCOTT            3000
        10 KING             5000
        20 FORD             3000
cs



● 직급(JOB)이 MANAGER인 사람의 속한 부서의 부서 번호와 부서명과 지역을 출력 해보자

1
2
3
4
5
6
7
8
SQL> SELECT deptno, dname, loc FROM dept
  2  WHERE deptno IN(SELECT deptno FROM emp WHERE job='MANAGER' );
 
    DEPTNO DNAME          LOC
---------- -------------- -------------
        10 ACCOUNTING     NEW YORK
        20 RESEARCH       DALLAS
        30 SALES          CHICAGO
cs


● >ANY 예제


1
2
3
4
5
6
7
8
9
10
11
12
13
14
SQL> SELECT ename, sal FROM emp
  2  WHERE sal >ANY(SELECT sal FROM emp WHERE job='SALESMAN');
 
ENAME             SAL
---------- ----------
KING             5000
FORD             3000
SCOTT            3000
JONES            2975
BLAKE            2850
CLARK            2450
ALLEN            1600
TURNER           1500
MILLER           1300
cs


서브 쿼리를 보자, SELECT sal FROM emp WHERE job='SALESMAN'

SALESMAN을 기준으로 월급(sal)의! 최소값(>ANY)을 조건(WHERER)로 하였다


1
2
3
4
5
6
7
8
SQL> SELECT ename, sal FROM emp WHERE job='SALESMAN';
 
ENAME             SAL
---------- ----------
ALLEN            1600
WARD             1250
MARTIN           1250
TURNER           1500
cs

그렇다면 SALESMAN의 최소값을 확인 해보자

1250이다.

그렇기에 1250을 최소값으로 반환 하여

1250보다 높은 SAL들이 나온 것을 확인 할 수 있다



※예제 수정요망 ㅠㅠ

'DATABASE > ORACLE' 카테고리의 다른 글

[Oracle] Transaction  (3) 2018.04.21
[Oracle] SQL DML / DDL  (2) 2018.04.21
[Oracle] SQL Outer Join / ANSI Join  (1) 2018.04.20
[Oracle] SQL Self Join  (0) 2018.04.20
[Oracle] SQL JOIN  (0) 2018.04.20