데이터 형 변환 함수 (이채남 선생님 저 '오라클 실습' 참고)

2016. 1. 28. 22:55

-- 데이터형 변환함수

-- 기본 사항              

--          날짜 ---> 문자 ---> 숫자

--             TO_CHAR   TO_NUMBER


--          날짜 <--- 문자 <--- 숫자

--             TO_DATE   TO_CHAR


--          날짜  --/--> 숫자 (불가능)

--          날짜 <--/--  숫자 (불가능)


-- TO_CHAR(1)

-- 날짜를 문자로

-- 기본예제

select to_char(sysdate,'YYYY-MM-DD HH24:MI:SS') from dual; --2016-01-28 21:37:07

select to_char(sysdate,'YyyY') from dual; --2016

select to_char(sysdate,'YYY') from dual; --016

select to_char(sysdate,'YY') from dual; --16

select to_char(sysdate,'Y') from dual; --6

select to_char(sysdate,'SYEAR') from dual; -- TWENTY SIXTEEN

select to_char(sysdate,'YEAR') from dual; --TWENTY SIXTEEN

select to_char(sysdate,'Q') from dual; --1

select to_char(sysdate,'mm') from dual; --01

select to_char(sysdate,'month') from dual; --1월(January)

select to_char(sysdate,'mon') from dual; --1월(JAN)

select to_char(sysdate,'D') from dual; --5

select to_char(sysdate,'DD') from dual; --28

select to_char(sysdate,'DDD') from dual; --028

select to_char(sysdate,'DAY') from dual; --목요일 (Thursday)

select to_char(sysdate,'DY') from dual; --목 (THU)

select to_char(sysdate,'AM') from dual; --오후

select to_char(sysdate,'PM') from dual; --오후

select to_char(sysdate,'A.M.') from dual; --오후

select to_char(sysdate,'P.M.') from dual; --오후

select to_char(sysdate,'HH') from dual; --09

select to_char(sysdate,'HH24') from dual; --21

select to_char(sysdate,'MI') from dual; --37

select to_char(sysdate,'SS') from dual; --07


--응용

select to_char(sysdate,'DD') from dual; --28 --> 원본

select to_char(sysdate,'DDTH') from dual; --28TH (서수)

select to_char(sysdate,'DDSP') from dual; --TWENTY-EIGHT (숫자를 철자로)

select to_char(sysdate,'DDSPTH') from dual; --TWENTY-EIGHTH (서수를 철자로)

select to_char(sysdate,'DDTHSP') from dual; --TWENTY-EIGHTH (서수를 철자로2)


--따라하기

select to_char(sysdate,'yy/mm/dd') date1, --16/01/28

to_char(sysdate,'yyyy.mm.dd') date2, --2016.01.28

to_char(sysdate,'month dd.yyyy') date3, --1월 28.2016

to_char(sysdate,'DY DD MON YY') date4, --목 28 1월 16

to_char(sysdate,'Day Mon DD') date5 --목요일 1월 28

from dual;


-- 연습문제

select emp_name, emp_id, birth_date from temp; --> 생일과 요일 출력변환

select emp_name, emp_id, to_char(birth_date,'yyyy "년" month dd"일" Day') from temp; --> 정답



-- TO_CHAR(2)

-- 숫자를 문자로

select to_char(192939,'999999') from dual; --192939

select to_char(192939,'00000000') from dual; --00192939

select to_char(192939,'999,999') from dual; --192,939

select to_char(192.939,'999.99') from dual; --192.94

select to_char(192939,'$999999') from dual; --$192939

select to_char(192939,'L999,999') from dual; --         ₩192,939

select to_char(-192939,'999999MI') from dual; --192939- (마이너스표시)

select to_char(192939,'999999EEEE') from dual;   2E+05 (과학적표기)


--따라하기

select to_char(1234,'09,999') number1, --01,234

       to_char(1234.56,'99,999.99') number2, --1,234.56

       to_char(1234,'$99,999') number3, --$1,234

       to_char(1234,'L99,999') number4, --         ₩1,234

       to_char(-1234,'99,999MI') number5 --1,234-

       -- to_char(-1234,'99,999EEEE')

       from dual;

       

--연습문제

select salary,

       to_char(salary, '099,999,999') SAL1,

       to_char(salary, '999,999,999.00') SAL2, 

       to_char(salary, '$999,999,999') SAL3,

       trim(to_char(salary, 'L999,999,999')) SAL4,

       to_char(salary, '999,999,999') SAL5 ,

       to_char(salary, '99EEEE') SAL6

       from temp;


       

--TO_NUMBER

--문자를 숫자로(O)

--날짜를 숫자로(X)


--따라하기

select to_number('123456') TONUM1, --123456

       to_number('123,456','999999') TONUM2 --123456

       from dual;


       

--TO_DATE

--문자를 날짜로(O)

--숫자를 날짜로(X)


select to_date('1997-12-31 13:33:44','yyyy-mm-dd hh24:mi:ss')

from dual; -- 97/12/31


select round(sysdate - to_date('19700101','yyyy-mm-dd')) "며칠"

from dual; --16829


--1970년 1월 1일 00시00분00초에서 946075441초가 지나면??

select to_date('19700101000000','yyyymmddhh24miss') +

           946075441 / (24*60*60)

           from dual; -- 99/12/24

곰돌곰둘 Oracle_DB_Unix admin/미분류

날짜 함수 (이채남 선생님 오라클 실습 2장 중)

2016. 1. 27. 23:06

-- 1. date type + date type 의 가산은 불가능

select sysdate + birth_date

from temp

where emp_name = '홍길동';


-- 2. date type - date type 의 감산은 가능

select emp_name, sysdate-birth_date from temp

where emp_name = '홍길동';


-- 3. date type에 number type을 더하거나 빼서 날짜 계산이 가능

select emp_name, birth_date, birth_date + 1.5, birth_date - 1.5

from temp

where emp_name = '홍길동';


-- * sysdate는 초단위까지 계산되지만 기본적으로 년/월/일만 출력됨

select sysdate from dual;


-- 4. date type 출력 형식 지정 및 시, 분, 초 계산

select to_char(sysdate,'HH24:MI:SS') AS TIME,

       to_char(sysdate + 14/24, 'HH24:MI:SS') AS "14시간 후",

       to_char(sysdate + 30 / (24*60) , 'HH24:MI:SS') AS "30분후",

       to_char(sysdate + 50 / (24*60*60), 'HH24:MI:SS') AS SEC from dual;

       

-- 5. 날짜 함수의 활용 (1) ADD_MONTHS

select sysdate,

       add_months(sysdate,1), -- 한달 더하기

       add_months(sysdate,-13) -- 13개월 빼기

from   dual;


-- 6. 날짜 함수의 활용 (1) MONTHS_BETWEEN

select months_between(sysdate,birth_date) MON_TERM

from  temp

where emp_name = '홍길동';


-- 7. 연습문제 : ADD_MONTH 함수의 사용

select emp_name, round(months_between(birth_date,sysdate)) MONTHS,

       to_char(birth_date,'yyyy-mm-dd') BIRTH,

       to_char(add_months(sysdate,months_between(birth_date,sysdate)),'yyyy-mm-dd') TO_YE1,

       to_char(add_months(sysdate,trunc(months_between(birth_date,sysdate))),'yyyy-mm-dd') TO_YE2,

       to_char(add_months(sysdate,round(months_between(birth_date,sysdate))),'yyyy-mm-dd') TO_YE3 from temp;

       

-- 8. LAST_DAY --> 해당 월의 말일 산출

select emp_name, birth_date, last_day(birth_date)

from temp;

곰돌곰둘 Oracle_DB_Unix admin/미분류

데이터 형 변환이 일어나는 예

2015. 10. 25. 21:50

쿼리를 작성할 때 사실 잘 의식하지는 못하는 부분이지만, 오라클이 query를 수행하는 과정에서 형 변환은 자주, 끊임없이 일어나고 있다. 이채남 저 '오라클 실습'에서 소개한 예제를 여기에 정리하고자 한다.


형변환 사례 1) CONCAT 또는 합성연산자 ||의 사용 시


select emp_id || emp_name from temp;

CONCAT또는 합성연산자는 두개 이상의 문자열을 하나의 문자열로 묶어주는 함수이다.

emp_id가 number type이기 때문에, 이때 오라클은 자동으로 number --> 문자(char? varchar2?)로 형변환을 시켜준다. 즉 위의 쿼리는 다음과 같이 처리가 되게 된다.

(1) 형변환(숫자 to 문자) --> (2)문자열 합성


형변환 사례 2) number type에 문자열 함수를 사용할 경우

위의 사례보다 더 중요하게 생각되는 사례 인데, 예를 들어 다음 쿼리를 보자.


select emp_id, emp_name

from temp

where substr(emp_id,1,4) + 1 = 1998;

emp_id의 첫번째 4자리에 1을 더한 값이 1998인 사원의 사번과 이름을 출력하는 쿼리인데, 문제는 emp_id는 number type인데 substr은 문자열 함수라는 것이다.

따라서 오라클은 위의 쿼리를 수행하기 위해 다음과 같이 형변환을 수행한다.


(1)형변환(숫자 to 문자) -->            (2) substr함수 수행 -->  (3)형변환 (문자 to 숫자) --> (4) 연산

19970101(숫자) --> 19970101(문자)  -->   1997(문자)  -->  1997(숫자)     --> +1 연산 --> 1998(숫자)


두번의 형변환이 일어났는데, 두번째 형변환은 당연히 +1이라는 연산을 수행해주기 위해 일어난 것이다. 비교 대상인 값인 =1998은 숫자로 쓰인 것이므로 세번째 형변환은 일어나지 않는다. 만약 쿼리를 다음과 같이 썻다면, 세번째 형 변환도 일어날 것이다.


select emp_id, emp_name

from temp

where substr(emp_id,1,4) + 1 = '1998';


정리해보니 사실 사례 1과 2는 같은 사례였구나..

이 외에도 형변환의 사례를 찾는다면 이 포스팅을 계속 업데이트 하고 싶은 바람이다.

곰돌곰둘 Oracle_DB_Unix admin/미분류

TABLE에 대한 사소한 몇 가지의 의문 해결

2015. 10. 25. 21:26

-- 첫번째 의문 : NOT NULL과 PRIMARY KEY

궁금증 1) NOT NULL 제약조건과 PRIMARY KEY 제약조건을 같이 쓸수 있을까? --> YES


아래 DDL문은 이채남 저 오라클 실습 1장에 나오는 예제 테이블 생성 쿼리이다. EMP_ID에 NOT NULL과 PRIMARY KEY 제약조건을 같이 줬는데, PK 자체가 NOT NULL에 Unique니까 이렇게 잡아주는게 의미가 있나? 또는 틀린 구문은 아닐까? 라는 의문이 들었다.


CREATE TABLE TEMP2 (

 EMP_ID      NUMBER NOT NULL PRIMARY KEY,

 EMP_NAME    VARCHAR2(10) NOT NULL,

 BIRTH_DATE  DATE,

 DEPT_CODE   VARCHAR2(06) NOT NULL,

 EMP_TYPE    VARCHAR2(04),

 USE_YN      VARCHAR2(01) NOT NULL,

 TEL         VARCHAR2(15),

 HOBBY       VARCHAR2(30),

 SALARY      NUMBER,

 LEV         VARCHAR2(04)

);


아래 쿼리를 수행시켜서 확인해봤다.

select a.owner, a.constraint_name, a.constraint_type, a.table_name, b.column_name, a.search_condition from user_constraints a, user_cons_columns b

where a.owner=b.owner

and a.constraint_name=b.constraint_name

and a.table_name=b.table_name

and a.table_name='TEMP2';


OWNER   CONS_NAME CONS_TYPE TAB_NAME COL_NAME SEARCH_CONDITION

PRAC01 SYS_C0011311 C TEMP2 EMP_ID "EMP_ID" IS NOT NULL

PRAC01 SYS_C0011312 C TEMP2 EMP_NAME "EMP_NAME" IS NOT NULL

PRAC01 SYS_C0011313 C TEMP2 DEPT_CODE "DEPT_CODE" IS NOT NULL

PRAC01 SYS_C0011314 C TEMP2 USE_YN "USE_YN" IS NOT NULL

PRAC01 SYS_C0011315 P TEMP2 EMP_ID


위의 결과에서 EMP_ID 컬럼에 PK constraint 와 NOT NULL Constraint 조건이 함께 생성된걸 확인 할 수 있다.

만약 첫번째 DDL문에서 not null 조건을 뺀 채 생성한다면 어떻게 될까?


OWNER   CONS_NAME CONS_TYPE TAB_NAME COL_NAME SEARCH_CONDITION

PRAC01 SYS_C0011328 C TEMP2 EMP_NAME "EMP_NAME" IS NOT NULL
PRAC01 SYS_C0011329 C TEMP2 DEPT_CODE "DEPT_CODE" IS NOT NULL
PRAC01 SYS_C0011330 C TEMP2 USE_YN "USE_YN" IS NOT NULL
PRAC01 SYS_C0011331 P TEMP2 EMP_ID

위의 결과처럼 EMP_NAME에 대해 NOT NULL constraint는 생성되지 않는다.

궁금증 2) PRIMARY KEY  컬럼에 Null을 명시해도 오류가 나지 않는가? 테이블이 생성되는가? --> YES

CREATE TABLE TEMP2 (

 EMP_ID      NUMBER NULL PRIMARY KEY,

 EMP_NAME    VARCHAR2(10) NOT NULL,

 BIRTH_DATE  DATE,

 DEPT_CODE   VARCHAR2(06) NOT NULL,

 EMP_TYPE    VARCHAR2(04),

 USE_YN      VARCHAR2(01) NOT NULL,

 TEL         VARCHAR2(15),

 HOBBY       VARCHAR2(30),

 SALARY      NUMBER,

 LEV         VARCHAR2(04)

);


결과적으로 위와 같이 생성해도 아~무 문제도 없다. 언뜻 생각하면 헷갈릴 수 있지만 각 컬럼값에 대해 NULL이 default 값이란걸 생각해보면 당연한 결과이다. 즉 위의 쿼리는 아래의 쿼리와 동일하다.


CREATE TABLE TEMP2 (

 EMP_ID      NUMBER PRIMARY KEY,

 EMP_NAME    VARCHAR2(10) NOT NULL,

 BIRTH_DATE  DATE,

 DEPT_CODE   VARCHAR2(06) NOT NULL,

 EMP_TYPE    VARCHAR2(04),

 USE_YN      VARCHAR2(01) NOT NULL,

 TEL         VARCHAR2(15),

 HOBBY       VARCHAR2(30),

 SALARY      NUMBER,

 LEV         VARCHAR2(04)

);


그리고 당연하게도 PK가 해당 컬럼에 있기 때문에, 테이블 생성 시 NULL을 명시했다고 해서 해당 컬럼이 NULL인 레코드가 테이블에 insert되는 경우는 발생하지 않기 때문에 안심하시라.


궁금증 3)  NOT NULL(NULL), PRIMARY KEY 구문의 순서가 바뀌어도 되는가? --> YES

실험으로 확인했으나, 간단한 부분이니 따로 결과를 올리지는 않았다.


-- 두번째 의문 : DROP TABLE 구문

DROP TABLE구문은 다음과 같다.

DROP TABLE schema_name.table_name [cascade constraints] [purge];


궁금증 4) cascade constraints 구문

일단 헷갈리는 부분인 Foreign Key에 대해 정리하고 넘어가자. Foreign Key 생성 예제는 다음과 같다.

ALTER TABLE employees

ADD CONSTRAINTS fk_deptno FOREIGN KEY (department_id)

REFERENCES departments (department_id);


여기서 자식 테이블은 employees, 부모 테이블은 departments가 된다. 즉, Foreign Key constraint(이하 Foreign Key)는 자식테이블에 생성되는 것이다. 또한 Foreign Key는 referential integrity constraint(참조 무결성 제약조건)과 같은 말이다.

도표로 표현하면 다음과 같다.


자식테이블                                     부모테이블

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

Foreign Key  --> refers to (참조)  -->   Primary Key or Unique Key

                                           *referential integrity relationship (참조 무결성 관계) 


다시 DROP TABLE 구문으로 돌아가서, cascade constraints구는 만약 drop될 테이블이 부모테이블로서 참조되고 있다면, 자식테이블의 참조하는 Foreign Key 제약조건을 함께 삭제하라는 명령이다. 왜냐하면 자식 테이블의 외래키에 의해 참조되고 있는 테이블은 Drop 될 수 없기 때문이다.

결국 drop할 테이블이 부모 테이블이 아니라면 cascade constraints 구문은 불필요하다.


사족) Foreign Key 제약조건과 관련하여 끊임없이 헷갈리는 이유는 짧은 경험 때문이기도 하지만, 자식테이블에 FK를 생성할 때 부모테이블에 대응되는 무언가(?)가 없기 때문이지 않나... 라는 생각이 든다. 따라서 "자식테이블에 FK를 생성하면, 부모 테이블엔 referential integrity relationship(참조 무결성 관계)가 생긴다." 정도로 정리를 해두려고 한다. 물론 엄밀히 따지면 부-자 테이블 간에 해당 관계가 성립되는 것이겠지만...



궁금증 5) Purge 구문과 Constraints

아시다시피 purge 구문은 oracle 10g의 신 기능인 recycle bin과 관련된 구문이다. 즉 테이블을 drop 했을 때 purge하기 전까지는 해당 테이블은 recycle bin에 place되어 복구가 가능하다. 그런데 constraints와 관련하여..

일단 테이블 드랍

drop table temp2;


그리고 다음 구문으로 조회를 한다.

select a.owner, a.constraint_name, a.constraint_type, a.table_name, b.column_name, a.search_condition from user_constraints a, user_cons_columns b

where a.owner=b.owner

and a.constraint_name=b.constraint_name

and a.table_name=b.table_name

and a.table_name like 'BIN%';


OWNER   CONS_NAME CONS_TYPE TAB_NAME COL_NAME SEARCH_CONDITION

PRAC01 BIN$IFoPPCQFRgiZ37xFqNreYA==$0 C BIN$rcwgjLNxTnyUVLV7CgSwgw==$0 EMP_ID "EMP_ID" IS NOT NULL

PRAC01 BIN$VQtSPYN1SoWRJruYXqAABg==$0 C BIN$rcwgjLNxTnyUVLV7CgSwgw==$0 EMP_NAME "EMP_NAME" IS NOT NULL

PRAC01 BIN$za5xbIlVT9u3ju49LJEoDA==$0 C BIN$rcwgjLNxTnyUVLV7CgSwgw==$0 DEPT_CODE "DEPT_CODE" IS NOT NULL

PRAC01 BIN$JRARQl33RA2Yup/70nI8eQ==$0 C BIN$rcwgjLNxTnyUVLV7CgSwgw==$0 USE_YN "USE_YN" IS NOT NULL

PRAC01 BIN$33sqjnJjRJqWelc3yo2tkg==$0 P BIN$rcwgjLNxTnyUVLV7CgSwgw==$0 EMP_ID


다음과 같이 constraints도 완전히 삭제 되지 않고 replace 되었음을 확인할 수 있다.

그리고 recycle bin을 purge 한다.

purge recyclebin;


다시 조회하면 recycle bin에 replaced되었던 constraints도 삭제 되었음을 확인할 수 있다.

만약 해당 테이블을 복원할 필요가 없거나, 백업이 잘 되어 있어서 recycle bin 보관이 필요치 않은 경우라면, 다음 구문으로 깔끔하게 constraints 까지 정리하는 편이 좋겠다.


drop table temp2 purge;


곰돌곰둘 Oracle_DB_Unix admin/미분류

실습 56. 리스트 파티션 테이블 생성과 관리

2015. 10. 13. 01:08

※ 책에서 제시하는 실습용 스키마를 생성하지 않았기 때문에 다소 예제가 다르나, 내용은 대동소이합니다. SCOTT 유저의 기본 스키마 (EMP, departments) 등을 이용합니다.


(1)파티션 테이블 생성

create table scott.emp_list_part (

EMPNO NUMBER(22),

ENAME VARCHAR2(10),

JOB VARCHAR2(9),

MGR NUMBER(22),

HIREDATE DATE,

SAL NUMBER(22),

COMM NUMBER(22),

DEPTNO NUMBER(22),

constraint pk_list_empno primary key(empno)

using index tablespace indx

)

partition by list (job)

(

partition job_entry values ('CLERK','ANALYST') tablespace t1,

partition job_middle values ('MANAGER','SALESMAN') tablespace t2,

partition job_high values ('PRESIDENT') tablespace t3);


(2)데이터 삽입

insert into emp_list_part

select * from emp;


commit;


(3)파티션 등 확인

select * from user_tab_partitions

where table_name = 'EMP_LIST_PART';


select table_name, partitioned from user_tables;


select name, column_name

from user_part_key_columns

where name = 'EMP_LIST_PART';


(4)파티션 드랍 (alter table을 사용)

alter table emp_list_part

drop partition job_entry;


(5)데이터 재 삽입 --> 오류가 발생한다. partition key가 list에 존재하지 않기 때문!

insert into emp_list_part

select * from emp

where job in ('CLERK','ANALYST');

ORA-14400: inserted partition key does not map to any partition


(6)파티션 추가 (역시 alter table을 사용)

alter table emp_list_part

add partition job_entry values ('CLERK','ANALYST') tablespace t1;


(7)데이터 재 삽입 --> 어랏? 에러 다시 발생

insert into emp_list_part

select * from emp

where job in ('CLERK','ANALYST');


SQL 오류: ORA-01502: 인덱스 'SCOTT.PK_LIST_EMPNO'또는 인덱스 분할영역은 사용할 수 없은 상태입니다

01502. 00000 -  "index '%s.%s' or partition of such index is in unusable state"

*Cause:    An attempt has been made to access an index or index partition

           that has been marked unusable by a direct load or by a DDL

           operation

*Action:   DROP the specified index, or REBUILD the specified index, or

           REBUILD the unusable index partition


무엇이 문제인지 확인해본다.


dba_objects 조회 --> 정상


select owner, object_name, object_type, status from dba_objects

where owner = 'SCOTT'

and object_type= 'INDEX';


SCOTT PK_LIST_EMPNO INDEX VALID

SCOTT PK_EMP INDEX VALID

SCOTT PK_DEPT INDEX VALID


dba_indexes 조회 --> UNUSABLE로 문제가 생긴걸 알 수 있다.


select owner, index_name, status from dba_indexes

where owner = 'SCOTT';


SCOTT PK_DEPT VALID

SCOTT PK_EMP VALID

SCOTT PK_LIST_EMPNO UNUSABLE


-- 그럼 일단 REBUILD를 해준다.

alter index PK_LIST_EMPNO rebuild;


(8)데이터 재 삽입

insert into emp_list_part

select * from emp

where job in ('CLERK','ANALYST');


commit;

--> 인덱스 REBUILD 후에는 정상적으로 데이터가 삽입됨을 알 수 있다.


-- 이상의 실습에서 알 수 있는점

1. 새로운 레코드가 파티션 테이블에 insert 될 때, 오라클이 자동적으로 key column에 따라 각 파티션으로 배분해준다. 이때 key column 값이 list 에 없는 값이라면 ORA-14400 에러가 발생하며 값이 insert 되지 않는다.


2. (1)번 과정에서 인덱스를 비파티션 인덱스로 만들었다. 그 후 각 테이블 파티션(이 예제에서는 job_entry의 drop)에 변경이 발생하자 index가 unusable 상태로 변경된걸 확인할 수 있었다. (PK 제약조건이 딸린 INDEX가 unusable 상태로 변경되었기 때문에 새 레코드의 insert가 불가능) --> 이후 Rebuild를 통해 인덱스를 valid 상태로 만든 후에야 레코드의 insert가 가능했다.


즉, 파티션 테이블에 비파티션 인덱스가 구성되어 있다면, 특정 테이블 파티션에 변경(split, exchange, drop 등)을 가하고 난 후에는 반드시 INDEX를 REBUILD 해 주어야 한다.


--> 이런 불편함을 감소시키기 위해서 로컬 파티션 인덱스를 사용할 수 있다.


실습 재개

(9) 인덱스 삭제 (PK_LIST_EMPNO)

alter table emp_list_part

drop constraint pk_list_empno;

(* PK Constraint에 해당하는 index는 drop 할 수 없으므로, 대신 PK Constraint를 삭제 해준다. 그럼 자동적으로 해당 INDEX도 DROP 된다. 


참고

drop index PK_LIST_EMPNO

오류 보고 -

SQL 오류: ORA-02429: 고유/기본 키 적용을 위한 인덱스를 삭제할 수 없습니다.

02429. 00000 -  "cannot drop index used for enforcement of unique/primary key"

*Cause:    user attempted to drop an index that is being used as the

           enforcement mechanism for unique or primary key.

*Action:   drop the constraint instead of the index.)


(10) 로컬 파티션 인덱스 생성 --> 에러발생!


alter table emp_list_part add constraint pk_list_empno primary key(empno)

using index local tablespace indx;


SQL 오류: ORA-14039: 열을 분할영역한 것은 UNIQUE 인덱스로 키 열의 부분 집합을 폼 합니다

14039. 00000 -  "partitioning columns must form a subset of key columns of a UNIQUE index"

*Cause:    User attempted to create a UNIQUE partitioned index whose

           partitioning columns do not form a subset of its key columns

           which is illegal

*Action:   If the user, indeed, desired to create an index whose

           partitioning columns do not form a subset of its key columns,

           it must be created as non-UNIQUE; otherwise, correct the

           list of key and/or partitioning columns to ensure that the index'

           partitioning columns form a subset of its key columns


구문은 위와 같으나 에러가 발생했다. 그 이유는 local partition index의 대상 컬럼은 파티션 키 컬럼(여기서는 job)과 같아야 하기 때문이다.

empno를 조회 조건으로 많이 사용하기 때문에(가정) empno를 버릴 수는 없다. 따라서 다음과 같이 수정하여 인덱스를 생성하였다.


(11) 로컬 파티션 인덱스 생성 (local non-prefixed index) --> 성공

* 첫번째 컬럼이 empno, 즉 list partitioning key column이 아니기에 non-prefixed가 된다.


alter table emp_list_part add constraint pk_list_empno primary key(empno, job)

using index local tablespace indx;


생성 확인) DBA_OBJESTS에서는 생성이 된걸 확인할 수 있으나, DBA_SEGMENTS에서는 확인되지 않는다.

이 때 SEGMENT_TYPE을 INDEX PARTITION으로 해야 조회가 가능하다.

*SEGMENT_TYPE이 INDEX가 아닌 INDEX PARTITION인 까닭 : LOCAL PARTITION INDEX라는 개념 자체가 인덱스 파티션과 테이블 파티션이 1:1 매칭이 되는 걸 전제로 하고 있기 때문이다.


추가 확인) INDEX 파티션 생성여부, 구성 컬럼, Constraint 조건 확인


select * from dba_ind_partitions --> 인덱스 파티션은 여기서 확인할 수 있다.

where index_owner = 'SCOTT';


select * from dba_ind_columns

where index_owner = 'SCOTT';


select * from dba_constraints

where constraint_name = 'PK_LIST_EMPNO';


(12)테스트: 테이블 파티션 DROP

실제로 local index를 사용한 경우, 테이블 파티션 변경시에도 INDEX가 VALID로 유효한지 확인해보자.


alter table emp_list_part

drop partition job_entry; --> 테이블 파티션 DROP


인덱스 상태확인)

select owner, object_name, object_type, status from dba_objects

where owner = 'SCOTT'

and object_type= 'INDEX';


SCOTT PK_LIST_EMPNO INDEX VALID

SCOTT PK_EMP INDEX VALID

SCOTT PK_DEPT INDEX VALID

--> dba_objects 확인. 일단 정상이다


select owner, index_name, status from dba_indexes

where owner = 'SCOTT';


SCOTT PK_DEPT VALID

SCOTT PK_EMP VALID

SCOTT PK_LIST_EMPNO N/A

--> dba_indexes 확인. 상태가 N/A이다. (아까전 비 파티션 인덱스 때 UNUSABLE이었던것과 비교해보자)


select index_name, partition_name, status from dba_ind_partitions

where index_owner = 'SCOTT';


PK_LIST_EMPNO JOB_MIDDLE USABLE

PK_LIST_EMPNO JOB_HIGH USABLE

--> 나머지 인덱스 파티션의 경우 상태가 USABLE임을 확인 할 수 있다.


우선 PK Constraints가 정상인지 먼저 확인 해보자.


insert into emp_list_part

select * from emp

where job in ('MANAGER'); --> 일부러 중복되는 값을 insert

SQL 오류: ORA-00001: 무결성 제약 조건(SCOTT.PK_LIST_EMPNO)에 위배됩니다


--> PK Constraints가 정상적으로 작동함을 확인 할 수 있다.


delete emp_list_part

where empno='7499'

and job='SALESMAN';

commit; --> INSERT가 정상적으로 되는지 확인하기 위해 일부러 값을 삭제함.


insert into emp_list_part

select * from emp

where empno='7499'

and job='SALESMAN';

commit; --> INSERT가 정상적으로 되며, 따라서 PK 제약조건이 딸린 INDEX도 문제 없이 사용가능함을 확인하였다.

최종적으로 DROP 했던 테이블 파티션을 다시 추가해주고, 해당 파티션에 속했던 record들이 정상적으로 INSERT되는지 확인해보자.


(13) 파티션 추가 및 레코드 INSERT

alter table emp_list_part

add partition job_entry values ('CLERK','ANALYST') tablespace t1; --> 파티션 추가 완료


*추가 후 dba_objects, dba_ind_partitions에서 STATUS가 각각 valid, usable로 조회가 되나, 여전히 DBA_INDEXES에서는 N/A로 조회됨을 알수 있다.


insert into emp_list_part

select * from emp

where job in ('CLERK','ANALYST');

commit; --> 레코드 추가 완료


이로써 로컬 파티션 인덱스 생성시에는 특정 테이블 파티션에 대한 변경(drop, split, exchange 등)이 발생하더라도 해당 INDEX가 unusable 되지 않음을 확인하였다.


(참고) 

참고1.DBA_INDEXES의 STATUS에서 N/A는 무엇을 말하는 것일까? oracle reference를 읽어보자

--> Indicates whether a nonpartitioned index is VALID or UNUSABLE

nonpartition index일 경우에 대해서만 valid/unusable을 말해주고, partition index일 경우 N/A (해당 없음)이라는 것. partition index를 생성하면 초기값 부터가 N/A이다.


한편 ALL_IND_PARTITIONS의 STATUS 설명을 보면,

--> Indicates whether the index partition is usable (USABLE) or not (UNUSABLE)

즉 파티션 인덱스(=인덱스 파티션)은 dba_ind_partitions에서 조회하는 것이 맞다는 것이다.


참고2. 하지만 여전히 DBA_INDEXES 조회시 N/A로 나타나는게 찜찜하다... Rebuild를 하면 어떨까?

alter index PK_LIST_EMPNO rebuild;

오류 보고 -

SQL 오류: ORA-14086: 분할영역된 인덱스는 전체를 다시 만들 수 없습니다

14086. 00000 -  "a partitioned index may not be rebuilt as a whole"

*Cause:    User attempted to rebuild a partitioned index using

           ALTER INDEX REBUILD statement, which is illegal

*Action:   Rebuild the index a partition at a time (using

           ALTER INDEX REBUILD PARTITION) or drop and recreate the

           entire index


설명 처럼 파티션 인덱스는 개별적으로 (한번에 하나) rebuild 하여야 하며, 구문도 다르다. 예제 -->

alter index PK_LIST_EMPNO rebuild partition job_entry;


참고3. 파티션 인덱스 생성 구문

create index emp_list_part_idx1 on emp_list_part (job) local;

생각외로 간단하며 뒤에 local만 붙여주면 된다. 단 파티션 인덱스는 tablespace 미 지정시 해당 USER의 Default tablespace가 아니라 '짝지' 테이블 파티션과 동일한 tablespace에 위치하게 되므로, 이것이 싫을 경우 tablespace를 지정해주자. 구문은 다음과 같다.


create index emp_list_part_idx1 on emp_list_part (job) local

(partition list_p_idx1_entry tablespace indx,

partition list_p_idx1_middle tablespace indx,

partition list_p_idx1_high tablespace indx);


테이블 파티션과 인덱스 파티션의 매칭 결과는 어디서 볼 수 있는가? 역시 dba_ind_partitions에서 확인 가능하다.

SCOTT EMP_LIST_PART_IDX1 NO LIST_P_IDX1_HIGH 0 'CLERK', 'ANALYST'

SCOTT EMP_LIST_PART_IDX1 NO LIST_P_IDX1_ENTRY 0 'MANAGER', 'SALESMAN'

SCOTT EMP_LIST_PART_IDX1 NO LIST_P_IDX1_MIDDLE 0 'PRESIDENT'

SCOTT PK_LIST_EMPNO NO JOB_HIGH 0 'PRESIDENT'

SCOTT PK_LIST_EMPNO NO JOB_MIDDLE 0 'MANAGER', 'SALESMAN'

SCOTT PK_LIST_EMPNO NO JOB_ENTRY 0 'CLERK', 'ANALYST'


위에서부터 3번째까지 레코드를 보면 알겠지만, 초기 설계의도와는 다르게 매칭이 뒤죽박죽이 되었다... 인덱스 생성시 테이블 파티션을 지정해 줄수 없는가? --> 방법을 못 찾음... 찾으면 알려주시길 ㅠㅜ


내가 찾아낸 방법은 rename이 최선이었다. 구문은 다음과 같다 -->

alter index EMP_LIST_PART_IDX1 rename partition LIST_P_IDX1_HIGH to LIST_P_IDX1_ENTRY;

곰돌곰둘 Oracle_DB_Unix admin/(5)Self Training_오라클 관리

DATA TYPE - DATE 옆에는 Length를 쓰면 안돼요!

2015. 10. 12. 22:36

30분 동안 끙끙거린 문제...

왜 다음 문장이 오류가 발생하는가?


create table scott.emp_list_part (

EMPNO NUMBER(22),

ENAME VARCHAR2(10),

JOB VARCHAR2(9),

MGR NUMBER(22),

HIREDATE DATE(7),

SAL NUMBER(22),

COMM NUMBER(22),

DEPTNO NUMBER(22),

constraint pk_list_empno primary key(empno)

using index tablespace indx

)

partition by list (job)

(

partition job_entry values ('CLERK','ANALYST') tablespace t1,

partition job_middle values ('MANAGER','SALESMAN') tablespace t2,

partition job_high values ('PRESIDENT') tablespace t3);


--> 답은 DATE 때문이다. DATE TYPE 옆에는 LENGTH를 기재하지 않는다.


create table scott.emp_list_part (

EMPNO NUMBER(22),

ENAME VARCHAR2(10),

JOB VARCHAR2(9),

MGR NUMBER(22),

HIREDATE DATE,

SAL NUMBER(22),

COMM NUMBER(22),

DEPTNO NUMBER(22),

constraint pk_list_empno primary key(empno)

using index tablespace indx

)

partition by list (job)

(

partition job_entry values ('CLERK','ANALYST') tablespace t1,

partition job_middle values ('MANAGER','SALESMAN') tablespace t2,

partition job_high values ('PRESIDENT') tablespace t3);


수정완료.

곰돌곰둘 Oracle_DB_Unix admin/미분류

Oracle Database Administrator's Guide (1)

2015. 10. 11. 20:27

<Getting Started with Database Administration>


- SQL Plus를 사용하는 법 (connect remotely)

(1) Oracle DB 설치 (엔진 + Client)

(2) Oracle Client 설치

(3) Instant Client 설치


- Instant ID = System ID = SID


For a local connection, you identify the instance by setting operating system environment variables.

For a remote connection, you identify the instance by specifying a network address and a database 

service name. For both local and remote connections, you must set environment 

variables to help the operating system find the SQL*Plus executable and to provide the 

executable with a path to its support files and scripts. 


- environment variables

1)ORACLE_HOME

2)ORACLE_SID

3)ORACLE_HOME/bin

곰돌곰둘 Oracle_DB_Unix admin/(4)Oracle Manual

Oracle Manual 진도표

2015. 10. 11. 18:37

(1)Oracle Database Administrator's Guide

10/11(일) 47P ~ 51P

곰돌곰둘 Oracle_DB_Unix admin/미분류

[TIP] SELECT 절 서브쿼리 사용시 주의점

2015. 10. 8. 16:07

이렇게 수행 시 에러 발생함.


select a.employee_id, a.first_name||' '||a.last_name names, (select b.department_name

from employees a, departments b

where a.department_id = b.department_id) dep_names

from employees a

where a.department_id = 100;


ORA-01427: single-row subquery returns more than one row

01427. 00000 -  "single-row subquery returns more than one row"

*Cause:    

*Action:


--> 요렇게 고쳐서 수행해야 함


select a.employee_id, a.first_name||' '||a.last_name names, (select b.department_name

from departments b

where a.department_id = b.department_id) dep_names

from employees a

where a.department_id = 100;

곰돌곰둘 Oracle_DB_Unix admin/미분류

PL/SQL 서브프로그램

2015. 10. 8. 16:05

- 묻지마 블록(Anonymous block)의 경우 실행 시 마다 항상 컴파일을 수행해야 함. 이런 불편함을 줄이기 위해 1회 컴파일 후 반복해서 사용할 수 있는 블록을 오라클 객체로 제공함. --> 이것이 PL/SQL 서브 프로그램임.


- PL/SQL 서브프로그램 (PL/SQL subprogram)

  * 파리미터 고유의 이름을 가진 PL/SQL 블록

  * 내장 프로시져 및 함수를 통칭

  * 함수는 결과값을 반환하므로 SELECT나 WHERE 절 등에 위치 할 수 있음.


-- 함수 (내장 함수, 사용자 정의 함수, User Defined Function)


(예제1)

CREATE OR REPLACE FUNCTION hr.emp_salaries (emp_id number)

           RETURN NUMBER IS

    nSalaries NUMBER(9);        -- 여기까지 함수 명세부(specification)

BEGIN                           -- 여기서부터 구현부(body)


  nSalaries := 0;

   SELECT salary

     INTO nSalaries

     FROM hr.employees

    WHERE employee_id = emp_id;

    RETURN nSalaries;

  END;


(예제2)

SELECT 절에 함수 사용하기


select employee_id, first_name, emp_salaries(employee_id)

  from employees

 where department_id = 100;


-- 함수 활용하기

employees 테이블에서 각 사원의 부서명을 알고 싶을 경우

1) employees 테이블, departments 테이블 조인

2) select 절에 서브쿼리 사용 (서브쿼리 내에서는 조인이 사용됨)

3) FUNCTION 생성 후 사용


1. 함수만들기


create or replace function get_dep_name (dep_id number)

                 return varchar2 is

                 

               sDepName varchar2(30);

          begin

              select department_name

                into sDepName

                from departments

               where DEPARTMENT_ID = dep_id;

          

          RETURN sDepName;

END;


2. 사용하기


select employee_id, first_name||' '||last_name names, get_dep_name(department_id) dep_names

from employees

where department_id = 100;


--> 함수를 사용할 경우 조인이 발생하지 않는다!

이처럼 함수를 사용할 경우 성능상 이점이 발생하나, 함수 개수가 많아지는 등 관리상의 문제가 발생할 수 있으므로 경우에 따라 조인, 서브쿼리, 함수중에서 적절한 것을 택해 사용한다.


참고. 서브쿼리 사용시


select a.employee_id, a.first_name||' '||a.last_name names, (select b.department_name

from departments b

where a.department_id = b.department_id) dep_names

from employees a

where a.department_id = 100;



-- 프로시져 (내장 프로시져, Stored procedure)


(예제) 신입사원 등록 프로시져 만들기


create or replace procedure register_emp (

                                         f_name VARCHAR2, -- *자리수 명시 X

                                         l_name VARCHAR2,

                                         e_acct VARCHAR2,

                                         j_id   VARCHAR2)  IS

begin

  insert_into employees (employee_id, first_name, last_name, email, hire_date, job_id)

                 values (employee_seq.nextval, f_name, l_name, e_acct, sysdate, j_id);

commit;

  

exception when others then  

  dbms_output.put_line(f_name||' '||l_name||' register is failed!');

rollback;

end;


* 프로시저의 파라미터로 명시할 때는 VARCHAR2만 명시한다. 구체적인 자리수(BYTES)까지 명시하게 되면 오류가 발생한다.


create or replace procedure register_emp (

                                         f_name VARCHAR2,

                                         l_name VARCHAR2,

                                         e_acct VARCHAR2,

                                         j_id   VARCHAR2)  IS

begin

  insert into employees (employee_id, first_name, last_name, email, hire_date, job_id)

                 values (employees_seq.nextval, f_name, l_name, e_acct, sysdate, j_id);

commit;

  

exception when others then  

  dbms_output.put_line(f_name||' '||l_name||' register is failed!');

rollback;

end;


(예제) 프로시져 사용하기

exec register_emp ('조영', '대', 'djoyoung', 'IT_PROG');

곰돌곰둘 Oracle_DB_Unix admin/(3)뇌자극 오라클 PL/SQL