IT/Framework
Mybatis - Cursor is closed 해결법
Normal_One
2019. 7. 21. 16:46
Spring framework 기반으로 Mybatis 설정 완료 후 프로시져를 통해서 리스트 값을 받아올 때 종종 Cursor is closed라는 에러가 발생하는 것을 확인할 수 있습니다. exception으로 throw하기 때문에 난감한 상황이 됩니다. 그렇다고 DB툴을 켜서 프로시져를 실행해보면 정상적으로 응답이 나오는 것을 확인할 수 있습니다. Mybatis에서는 해당 상황이 발생할 때SqlException으로 던지기 때문인 듯 한데, 이를 해결해 주려면 Cursor를 프로시져 도입부에서 무조건 Select 해주면 됩니다. 다만, 값을 내려주면 안되는 상황이고 Select를 태울 필요가 없는데 처음부터 Select가 실행되면 문제가 되기도 하니 아래와 같이 해결하시면 됩니다.
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
|
CREATE OR REPLACE PROCEDURE TEST_PROCEDURE
(
O_CURSOR OUT SYS_REFCURSOR
) IS
V_ERR01 EXCEPTION;
COUNT NUMBER;
BEGIN
OPEN O_CURSOR FOR
SELECT NULL FROM DUAL WHERE ROWNUM = 0;
-- ROWNUM을 0으로 해줘야 NULL값이 커서에 생성되지 않는다.
-- Cursor를 명시적으로 무조건 Open하여 Cursor is Closed 방지
SELECT COUNT(SALARY)
INTO COUNT
FROM EMPLOYEES
WHERE SALARY > 30000;
EXCEPTION
WHEN NO_DATA_FOUND THEN
NULL;
WHEN OTHERS THEN
RAISE V_ERR01;
-- 말이 안되는 Exception이긴 하나 실제로 여기서 Exception이 발생하고
-- Cursor가 열리지 않은 상태로 Raise시 Mybatis에서 Cursor is Closed로 Exception을 던지는 걸 확인할 수 있다.
OPEN O_CURSOR FOR
SELECT SALARY FROM EMPLOYEES
WHERE SALARY > 30000;
END TEST_PROCEDURE;
|
cs |
정상적으로 돌아갈만한 프로시저는 아니긴한데 예제용이니까 이해해주시기 바라며, 요점은 Raise로 던지기 전에 무조건 Cursor를 한 번은 Open해야 한다는 것 입니다. 위에 적었듯이 Rownum을 0으로 한 이유는 그렇게 하지 않았을 시에 결과값을 받아올 때 Null값이 List 내부에 생기게 됩니다. 많은 도움이 되셨으면 합니다.