문돌이 존버/DB 및 SQL 스터디
SQLD ROWNUM, LIKE, UNION, INTERSECT, MINUS 예제
애뚱
2021. 2. 27. 22:28
반응형
예제 테이블 및 데이터 insertion
CREATE TABLE emp
(
empno NUMBER(4) NOT NULL,
ename VARCHAR2(10),
job VARCHAR2(9),
mgr NUMBER(4),
hiredate DATE,
sal NUMBER(7,2),
comm NUMBER(7,2),
deptno NUMBER(2)
);
ALTER TABLE emp ADD CONSTRAINT emp_pk PRIMARY KEY (empno);
INSERT INTO emp VALUES(7839, 'KING', 'PRESIDENT', NULL, TO_DATE('1981-11-17', 'yyyy-mm-dd'), 5000, NULL, 10);
INSERT INTO emp VALUES(7698, 'BLAKE', 'MANAGER', 7839, TO_DATE('1981-05-01', 'yyyy-mm-dd'), 2850, NULL, 30);
INSERT INTO emp VALUES(7782, 'CLARK', 'MANAGER', 7839, TO_DATE('1981-06-09', 'yyyy-mm-dd'), 2450, NULL, 10);
INSERT INTO emp VALUES(7566, 'JONES', 'MANAGER', 7839, TO_DATE('1981-04-02', 'yyyy-mm-dd'), 2975, NULL, 20);
INSERT INTO emp VALUES(7788, 'SCOTT', 'ANALYST', 7566, TO_DATE('1987-04-19', 'yyyy-mm-dd'), 3000, NULL, 20);
INSERT INTO emp VALUES(7902, 'FORD', 'ANALYST', 7566, TO_DATE('1981-12-03', 'yyyy-mm-dd'), 3000, NULL, 20);
INSERT INTO emp VALUES(7369, 'SMITH', 'CLERK', 7902, TO_DATE('1980-12-17', 'yyyy-mm-dd'), 800, NULL, 20);
INSERT INTO emp VALUES(7499, 'ALLEN', 'SALESMAN', 7698, TO_DATE('1981-02-20', 'yyyy-mm-dd'), 1600, 300, 30);
INSERT INTO emp VALUES(7521, 'WARD', 'SALESMAN', 7698, TO_DATE('1981-02-22', 'yyyy-mm-dd'), 1250, 500, 30);
INSERT INTO emp VALUES(7654, 'MARTIN', 'SALESMAN', 7698, TO_DATE('1981-09-28', 'yyyy-mm-dd'), 1250, 1400, 30);
INSERT INTO emp VALUES(7844, 'TURNER', 'SALESMAN', 7698, TO_DATE('1981-09-08', 'yyyy-mm-dd'), 1500, 0, 30);
INSERT INTO emp VALUES(7876, 'ADAMS', 'CLERK', 7788, TO_DATE('1987-05-23', 'yyyy-mm-dd'), 1100, NULL, 20);
INSERT INTO emp VALUES(7900, 'JAMES', 'CLERK', 7698, TO_DATE('1981-12-03', 'yyyy-mm-dd'), 950, NULL, 30);
INSERT INTO emp VALUES(7934, 'MILLER', 'CLERK', 7782, TO_DATE('1982-01-23', 'yyyy-mm-dd'), 1300, NULL, 10);
INSERT INTO emp VALUES(7940, 'X', 'CLERK', 7790, TO_DATE('1982-02-25', 'yyyy-mm-dd'), 1346, NULL, 10);
1. 부분 범위 처리
ROWNUM (Oracle)
# nest 구조문을 통해 미리 정렬한 뒤에 ROWNUM 적용해야 함
SELECT ROWNUM,
A.*
FROM (
SELECT *
FROM emp
ORDER BY hiredate
)A;
TOP (SQL Server=MS SQL)
# 나이 순서대로(내림차순) 정렬 후 탑 3 row 리턴
SELECT TOP 3 *
FROM TABLE
ORDER BY AGE DESC;
# 상위 50% row 리턴
SELECT TOP 50 PERCENT *
FROM TABLE
ORDER BY AGE DESC;
2. SQL 연산자 입력값을 비교하여 논리값 출력
LIKE '문자열'
# '%' -> 0개 이상 문자
SELECT *
FROM emp
WHERE ename LIKE '%';
# '_' -> 1개 단일 문자
SELECT *
FROM emp
WHERE ename LIKE '_';
# '%문자열%' -> 단순히 문자열이 포함되면 가져옴
SELECT *
FROM emp
WHERE ename LIKE '%IN%';
# '%문자열' -> 해당 문자열로 끝나는 경우만 가져옴
SELECT *
FROM emp
WHERE ename LIKE '%N';
# '문자열%' -> 해당 문자열로 시작하는 경우만 가져옴
SELECT *
FROM emp
WHERE ename LIKE 'J%';
3. 함수
날짜형 함수
SYSDATE: 현재 시각 출력 (년, 월, 일, 시, 분, 초)
EXTRACT: 날짜형 부분 추출
SELECT EXTRACT(YEAR FROM SYSDATE) AS YEAR
,EXTRACT(MONTH FROM SYSDATE) AS MONTH
,EXTRACT(DAY FROM SYSDATE) AS DAY
FROM DUAL;
NEXT_DAY(기준일자, 지정요일): 지정된 요일 첫 날짜 출력
SELECT SYSDATE
, NEXT_DAY(SYSDATE, 'FRI') AS next_day
FROM DUAL;
조건문 IF-THEN-ELSE 형태
1. DECODE(컬럼, 기준값1, 출력값1, 기준값2, 출력값2, ..., 기본값)
2. CASE WHEN 조건절1 THEN 출력값1 ... ELSE 기본값 END
SELECT A.ename, A.job, DECODE(job, 'ANALYST', '분석가', 'SALESMAN', '영업맨', '-') job_translation
FROM emp A;
SELECT A.ename, A.job,
CASE
WHEN sal >= 2500 THEN '고연봉'
WHEN sal >= 1500 THEN '저연봉'
ELSE '뭐지?'
END AS standard
FROM emp A;
# ELSE 생략 시 NULL 출력
SELECT A.ename, A.job,
CASE
WHEN sal >= 2500 THEN '고연봉'
WHEN sal >= 1500 THEN '저연봉'
END AS standard
FROM emp A;
집합연산자
UNION(합집합): 컬럼 수와 데이터 타입이 모두 동일한 테이블 간 연산
SELECT 컬럼명 FROM 테이블명 A WHERE 조건절 UNION SELECT 컬럼명 FROM 테이블명 WHERE 조건절;
# 테이블 emp2 생성
CREATE TABLE emp2
(
empno NUMBER(4) NOT NULL,
ename VARCHAR2(10),
job VARCHAR2(9),
mgr NUMBER(4),
hiredate DATE,
sal NUMBER(7,2),
comm NUMBER(7,2),
deptno NUMBER(2)
);
INSERT INTO emp2 VALUES(1234, 'JUNG', 'PRESIDENT', NULL, TO_DATE('1981-11-17', 'yyyy-mm-dd'), 5000, NULL, 10);
INSERT INTO emp2 VALUES(7698, 'JUN', 'MANAGER', 7839, TO_DATE('1981-05-01', 'yyyy-mm-dd'), 2850, NULL, 30);
INSERT INTO emp2 VALUES(5235, 'DOCO', 'MANAGER', 7839, TO_DATE('1981-06-09', 'yyyy-mm-dd'), 2450, NULL, 10);
INSERT INTO emp2 VALUES(7566, 'HYAMA', 'MANAGER', 7839, TO_DATE('1981-04-02', 'yyyy-mm-dd'), 2975, NULL, 20);
INSERT INTO emp2 VALUES(4577, 'BOND', 'ANALYST', 7566, TO_DATE('1987-04-19', 'yyyy-mm-dd'), 3000, NULL, 20);
INSERT INTO emp2 VALUES(7902, 'PAST', 'ANALYST', 7566, TO_DATE('1981-12-03', 'yyyy-mm-dd'), 3000, NULL, 20);
INSERT INTO emp2 VALUES(9893, 'GINGER', 'CLERK', 7902, TO_DATE('1980-12-17', 'yyyy-mm-dd'), 800, NULL, 20);
INSERT INTO emp2 VALUES(7499, 'LUKE', 'SALESMAN', 7698, TO_DATE('1981-02-20', 'yyyy-mm-dd'), 1600, 300, 30);
INSERT INTO emp2 VALUES(1960, 'PHILL', 'SALESMAN', 7698, TO_DATE('1981-02-22', 'yyyy-mm-dd'), 1250, 500, 30);
INSERT INTO emp2 VALUES(7654, 'JAY', 'SALESMAN', 7698, TO_DATE('1981-09-28', 'yyyy-mm-dd'), 1250, 1400, 30);
INSERT INTO emp2 VALUES(0796, 'MANNY', 'SALESMAN', 7698, TO_DATE('1981-09-08', 'yyyy-mm-dd'), 1500, 0, 30);
INSERT INTO emp2 VALUES(7876, 'HALEY', 'CLERK', 7788, TO_DATE('1987-05-23', 'yyyy-mm-dd'), 1100, NULL, 20);
INSERT INTO emp2 VALUES(7900, 'ALEX', 'CLERK', 7698, TO_DATE('1981-12-03', 'yyyy-mm-dd'), 950, NULL, 30);
INSERT INTO emp2 VALUES(1222, 'CLAIRE', 'CLERK', 7782, TO_DATE('1982-01-23', 'yyyy-mm-dd'), 1300, NULL, 10);
INSERT INTO emp2 VALUES(9753, 'Y', 'CLERK', 7790, TO_DATE('1982-02-25', 'yyyy-mm-dd'), 1346, NULL, 10);
아래의 경우, 테이블 emp와 emp2의 ename 컬럼은 겹치는 것이 없다. 따라서 UNION을 사용하든, UNION ALL을 사용하든 결과는 똑같을 것이다.
# 중복 제거
SELECT A.ename
FROM emp A
UNION
SELECT B.ename
FROM emp2 B;
# 중복 포함
# 정렬하지 않음 -> 그대로 갖다 붙임
SELECT A.ename
FROM emp A
UNION ALL
SELECT B.ename
FROM emp2 B;
INTERSECT(교집합)
SELECT 컬럼명 FROM 테이블명 A WHERE 조건절 INTERSECT SELECT 컬럼명 FROM 테이블명 B WHERE 조건절
MINUS(차집합) # EXCEPT(sql server)
SELECT 컬럼명 FROM 테이블명 A WHERE 조건절 MNIUS SELECT 컬럼명 FROM 테이블명 B WHERE 조건절
# 위에서 empno만 겹치게 테이블을 구성했음
SELECT A.empno
FROM emp A
INTERSECT
SELECT B.empno
FROM emp2 B;
# 전체 집합에서 교집합 뺀 부분
SELECT A.empno
FROM emp A
MINUS
SELECT B.empno
FROM emp2 B;
728x90
반응형