티스토리 뷰
얼마 전에 stored function과 trigger를 첨으로 사용해보았다.
엄청 어려운 수준으로 사용은 안 해봤지만, 정리하면 누군가에게 도움이 될 수 있지 않을까.. 하는 마음에
(하루만 지나면 기억 못 하는 나 자신을 위해서...)
stored function, trigger에 대하여 각각 정리해보려고 한다.
참고 URL
http://www.postgresqltutorial.com/postgresql-stored-procedures
https://www.postgresql.org/docs/current/sql-createfunction.html
참고 사이트에서는 stored function의 장단점에 대하여 다음과 같이 기술하고 있다.
장점
- application과 database 서버 간의 통신을 줄일 수 있다.
application에서 발생할 수 있는 여러 번의 SQL 호출을 stored function 호출 한 번으로 처리할 수 있다.
- application의 성능을 향상할 수 있다.
stored function은 미리 컴파일된 상태로 database에서 동작하기 때문에 application의 부담을 줄일 수 있다.
- 재사용이 편리하다.
stored function을 정의해두면, 어떤 application에서도 자유롭게 호출하여 사용할 수 있다.
단점
- 개발 속도가 느려질 수 있다.
application 개발 스킬과 다른 스킬이 필요하기 때문에 개발 속도가 느려질 수 있다.
- 버전 관리 및 디버깅이 어렵다.
- MySQL, Microsoft SQL과 같은 다른 database 서버에서 동작하지 않을 수 있다.
stored function을 어떤 상황에서 사용할지는 각자 판단해야 할 것 같다.
참고 사이트에 stored function의 기초부터 잘 설명이 되어 있기 때문에
여기서는 2가지 상황을 가정하여 function을 구현하였다.
1. A table의 데이터를 읽어와서 B table에 insert
CREATE OR REPLACE FUNCTION insert_A_to_B(cond_id bigint)
returns int
AS $$
DECLARE
result int := 0;
data_info record;
data_value boolean;
now_date timestamp := now();
BEGIN
for data_info IN select * from A where id > cond_id loop
raise notice '%, %', data_info.id, data_info.value;
if data_info.value = 'PASS' then
data_value := true;
else
data_value := false;
end if;
insert into B
(id, value, created, modified)
values
(data_info.id, data_value, now_date, now_date);
result := result+1;
end loop;
return result;
END; $$
LANGUAGE 'plpgsql';
위 예제는 단순한 사용법을 구현한 것이기 때문에 복잡한 조건을 포함하지 않았다.
- function에서 사용할 변수는 DECLARE에서 정의를 해야 한다.
- select 구문의 결과를 사용하기 위해서 record type을 사용하였다.
반복문과 조건문은 기존 언어처럼 다양하게 제공되고 있다.
반복문 : WHILE LOOP, FOR LOOP...
조건문 : IF/ELSEIF/ELSE, CASE/WHEN/ELSE
해당 stored function은 아래와 같이 호출하여 사용할 수 있다.
select insert_A_to_B(100);
2. A table에 data insert (data는 object array로 입력, function 안에서 B table의 값을 참고하기)
object를 parameter로 넘기기 위하여 아래와 같이 사용자 type을 정의한다.
create type a_type as (priority bigint, value varchar(32));
stored function은 다음과 같이 정의한다.
CREATE OR REPLACE FUNCTION insert_A_list(a_type[])
returns int
AS $$
DECLARE
result int := 0;
a_object a_type;
now_date timestamp := now();
BEGIN
foreach a_object IN array $1 loop
if exists(select 1 from B where value=a_object.value) then
insert into A
(priority, value, created, modified)
values
(a_object.priority, a_object.value, now_date, now_date);
result := result+1;
end if;
end loop;
return result;
END; $$
LANGUAGE 'plpgsql';
parameter로 정의 부분에 parameter의 이름은 정의하지 않고
loop 구문에서 $1로 해당 parameter를 사용하였다.
(나 같이 이름 정하는 게 너무 어려운 사람들에게 좋은 방식이다.)
- foreach 구문을 사용하여 parameter로 넘어온 array를 처리하였다.
- exists 구문을 사용하여 table B의 값을 체크하였다.
위의 function은 아래와 같은 형태로 호출할 수 있다.
select insert_keyword_list(ARRAY[(1, 'test1'), (2, 'test2'), (3, 'test3')]::a_type[]);
호출할 때 ARRAY [.......]::a_type []과 같은 형식을 사용하기 싫다면
아래와 같이 variadic을 사용하여 parameter 정의하면
CREATE OR REPLACE FUNCTION insert_A_list(variadic a_type[])
아래와 같은 형식으로 function을 호출할 수 있다.
select insert_keyword_list((1, 'test1'), (2, 'test2'), (3, 'test3'));
variadic 사용에 주의할 점이 있다면, parameter가 100개가 넘어가면 에러가 발생하게 된다.
'Development' 카테고리의 다른 글
[PostgreSQL] 한글 정렬 시 collate 옵션 사용하기 (0) | 2020.03.01 |
---|---|
[PostgreSQL] trigger 사용하기 (history table 만들기) (0) | 2019.09.12 |
[PostgreSQL] upsert(insert .. conflict on ..) 구문 사용하기 (2) | 2019.08.25 |
node, babel7, eslint, prettier 설정하기(2019.6.9) with Webstorm (2) | 2019.06.09 |
Docker 실습하기 03편 - node 서버 실행하기 (0) | 2019.05.01 |
- Total
- Today
- Yesterday
- 개리비숍
- 메이트북스
- 퇴근길인문학수업
- sethgodin
- 투자
- 가나출판사
- 사경인
- 블랙피쉬
- graphql
- 백상경제연구원
- 독서
- 강형욱
- 오건영
- 토니로빈스
- ebs다큐프라임
- 책리뷰
- 자기개발
- 재테크
- docker
- PostgreSQL
- aws 자격증
- 유발하라리
- 송희구
- 경제
- 서삼독
- 한빛비즈
- 알에이치코리아
- 더숲
- 인류3부작
- graphql-java
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | |||||
3 | 4 | 5 | 6 | 7 | 8 | 9 |
10 | 11 | 12 | 13 | 14 | 15 | 16 |
17 | 18 | 19 | 20 | 21 | 22 | 23 |
24 | 25 | 26 | 27 | 28 | 29 | 30 |