54일차 (1) - SQL (계층형 질의, 서브쿼리)

Yohan·2024년 5월 8일
0

코딩기록

목록 보기
74/157

오라클 계층형 쿼리

트리 구조와 같은 계층적 데이터를 쉽게 조회하고 조작할 수 있도록 지원하는 SQL 기능

START WITH

  • 계층의 첫 단계를 어디서 시작할 것인지의 대한 조건
  • START WITH에 써있는 조건은 CONNECT BY에 써있는 조건에 해당되지않아도 무조건 쓴다.

CONNECT BY

  • 상위 노드와 하위 노드 간의 관계를 정의
  • CONNECT BY PRIOR 자식 = 부모 -> 순방향 탐색
  • CONNECT BY 자식 = PRIOR 부모 -> 역방향 탐색

LEVEL

  • 계층에서 현재 레코드의 깊이를 나타낸다. 시작점을 레벨 1로 간주하고 각각의 하위 레벨에 대해 1씩 증가

ORDER SIBLINGS BY

  • 같은 레벨끼리의 정렬을 정함.

예시

SELECT 
    LEVEL AS LVL,
    LPAD(' ', 4*(LEVEL-1)) || emp_no || '(' || emp_nm || ')' AS "조직인원",
    A.dept_cd,
    B.dept_nm,
    A.emp_no,
    A.direct_manager_emp_no,
    CONNECT_BY_ISLEAF -- 가장 바깥에 있는 부분 : 1
FROM tb_emp A
JOIN tb_dept B
ON A.dept_cd = B.dept_cd
START WITH A.direct_manager_emp_no IS NULL -- START WITH : 어디서부터 펼칠 것인지
--START WITH A.EMP_NO = '1000000037'
CONNECT BY PRIOR A.emp_no = A.direct_manager_emp_no
ORDER SIBLINGS BY A.emp_no DESC
;
  • PRIOR 자식 = 부모 이므로 순방향 전개
  • 같은 레밸일 경우 같은 층
  • 같은 레밸에서 내림차순 정렬 적용
SELECT 
    LEVEL AS LVL,
    LPAD(' ', 4*(LEVEL-1)) || emp_no || '(' || emp_nm || ')' AS "조직인원",
    A.dept_cd,
    B.dept_nm,
    A.emp_no,
    A.direct_manager_emp_no,
    CONNECT_BY_ISLEAF
FROM tb_emp A
JOIN tb_dept B
ON A.dept_cd = B.dept_cd
--START WITH A.direct_manager_emp_no IS NULL -- START WITH : 어디서부터 펼칠 것인지
START WITH A.EMP_NO = '1000000037'
CONNECT BY A.emp_no = PRIOR A.direct_manager_emp_no
ORDER SIBLINGS BY A.emp_no DESC
;
  • 자식 = PRIOR 부모 이므로 역방향 전개
  • 이현정 위에는 이순표, 김회장 밖에 없으므로 데이터가 3개만 출력

서브쿼리

SQL 문장 내에서 다른 SQL 문장을 포함하는 구조

서브쿼리의 종류

단일 행 서브쿼리 (Single-row Subquery)

  • 조회 결과가 1건 이하
  • 서브쿼리가 하나의 행만 반환
  • 주로 비교 연산자(=, >, <, >=, <=, <>)와 함께 사용
-- 20200525에 받은 급여가 회사전체의 20200525 전체 평균 급여보다 높은 사원들의 정보(사번, 이름, 급여지급일, 받은급여액수) 조회

-- 회사전체 20200525 평균급여 계산
-- 그 평균보다 높은 사원 조회

SELECT
    E.emp_no,
    E.emp_nm,
    S.pay_de,
    S.pay_amt
FROM tb_emp E
JOIN tb_sal_his S
ON E.emp_no = S.emp_no
WHERE S.pay_de = '20200525'
    AND S.pay_amt >= (
        SELECT
            AVG(pay_amt)
        FROM tb_sal_his
        WHERE pay_de = '20200525'
    )
;

-- 20200525 회사전체 급여평균
SELECT
    AVG(pay_amt)
FROM tb_sal_his
WHERE pay_de = '20200525'
;

다중 행 서브쿼리 (Multi-row Subquery)

  • 서브쿼리의 조회 건수가 0건 이상인 것
  • 서브쿼리가 둘 이상의 행을 반환하는 경우
  • 다중 행 비교 연산자(IN, ANY, ALL, EXISTS, NOT EXISTS)와 함께 사용

다중행 연산자

  1. IN : 메인쿼리의 비교조건이 서브쿼리 결과중에 하나라도 일치하면 참
  • ex ) salary IN (200, 300, 400)
    250 -> 200, 300, 400 중에 없으므로 false
  1. ANY, SOME : 메인쿼리의 비교조건이 서브쿼리의 검색결과 중 하나 이상 일치하면 참
  • ex ) salary > ANY (200, 300, 400)
    250 -> 200보다 크므로 true
  • IN 과 = ANY는 같다.
  1. ALL : 메인쿼리의 비교조건이 서브쿼리의 검색결과와 모두 일치하면 참
  • ex ) salary > ALL (200, 300, 400)
    250 -> 200보다는 크지만 300, 400보다는 크지 않으므로 false
  1. EXISTS : 메인쿼리의 비교조건이 서브쿼리의 결과 중 만족하는 값이 하나라도 존재하면 참
  • EXISTS를 통해 값을 구하려는 것이 아니고 조건에 맞는 것이 존재하는지 여부만 파악하면되기때문에 EXISTS의 () 안 SELECT에는 무엇이 와도 상관없음
-- 한국데이터베이스진흥원에서 발급한 자격증을 가지고 있는
-- 사원의 사원번호와 사원이름과 해당 사원의 한국데이터베이스진흥원에서 
-- 발급한 자격증 개수를 조회

SELECT
    E.emp_no,
    E.emp_nm,
    COUNT(S.certi_cd) "자격증 개수"
FROM tb_emp E
INNER JOIN tb_emp_certi S
ON E.emp_no = S.emp_no
WHERE S.certi_cd IN ( -- IN은 = ANY와 같음
                        SELECT certi_cd
                        FROM tb_certi
                        WHERE issue_insti_nm = '한국데이터베이스진흥원'
)
GROUP BY E.emp_no, E.emp_nm -- 사원별로 해야되기 때문에 그룹핑
ORDER BY E.emp_no
;
  • EXISTS
-- 주소가 강남인 직원들이 근무하고 있는 부서정보를 조회 (부서코드, 부서명)

SELECT
    dept_cd,
    dept_nm
FROM tb_dept
WHERE dept_cd IN (100009, 100010)
;

SELECT
    dept_cd, emp_nm, addr
FROM tb_emp
WHERE addr LIKE '%강남%'
;


-- EXISTS의 ()안 SELECT에는 무엇이 와도 상관없음
-- -> 조건에 맞는 것이 존재하는지 여부만 파악하면되기때문에
SELECT
    dept_cd,
    dept_nm
FROM tb_dept D
WHERE EXISTS ( 
            SELECT
                D.dept_cd
            FROM tb_emp E
            WHERE addr LIKE '%강남%'
                AND D.dept_cd = E.dept_cd
)
;

다중 컬럼 서브쿼리 (Multi-column Subquery)

  • 서브쿼리의 조회 컬럼이 2개 이상인 서브쿼리
  • 비교 연산자와 함께 사용되며, 주로 연관 서브쿼리로 작성
-- 부서원이 2명 이상인 부서 중에서 각 부서의 
-- 가장 연장자의 사번과 이름 생년월일과 부서코드를 조회

SELECT 
    A.emp_no, A.emp_nm, A.birth_de, A.dept_cd, B.dept_nm
FROM tb_emp A
JOIN tb_dept B
ON A.dept_cd = B.dept_cd
WHERE (A.dept_cd, A.birth_de) IN ( -- WHERE와 IN안의 SELECT 갯수를 맞춰야함
                        SELECT 
                            dept_cd, MIN(birth_de)
                        FROM tb_emp
                        GROUP BY dept_cd
                        HAVING COUNT(*) >= 2
                    )
ORDER BY A.emp_no
;

인라인 뷰 서브쿼리

  • FROM절이나 JOIN에 쓰는 서브쿼리
  • 목적: 테이블의 양을 줄여놓고 시작하기위함
-- 각 사원의 사번과 이름과 평균급여정보를 알고 싶음
SELECT
    E.emp_no,
    E.emp_nm,
    AVG(S.pay_amt)
FROM tb_emp E
JOIN (
    SELECT emp_no, AVG(pay_amt) AS pay_avg
    FROM tb_sal_his
    GROUP BY emp_no
) S
ON E.emp_no = S.emp_no
ORDER BY E.emp_no
;

스칼라 서브쿼리 (SELECT, INSERT, UPDATE절에 쓰는 서브쿼리)

  • 성능을 고려해서 조인할 필요없이 뽑아올 수 있는 몇개 안되는 정보는 스칼라 서브쿼리 이용
-- 사원의 사번, 사원명, 부서명, 생년월일, 성별코드를 조회
SELECT
    E.emp_no,
    E.emp_nm,
    (SELECT dept_nm FROM tb_dept D WHERE E.dept_cd = D.dept_cd) AS dept_nm, -- 부서명
    E.birth_de,
    E.sex_cd
FROM tb_emp E
;
profile
백엔드 개발자

0개의 댓글