본문 바로가기

Oracle/PL

PL Procedure 프로시져, Cursor 커서

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
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
/*
    block 되어 있는 처리 -> 함수회
    매개변수, RETURN 값
    
    PROCUDURE   -- INSERT, DELETE, UPDATE
    
    FUNCTION    - SELECT
    
    TRIGGER     - UTIL  검사를 할때 사용
*/
SET SERVEROUTPUT ON
 
CREATE OR REPLACE PROCEDURE myProc(inNum IN NUMBER, outNum OUT NUMBER) --매개변수/ 리턴값 
IS
    -- 선언부
begin
    -- 실행부
    dbms_output.put_line('inNum:' || inNum);
    
    outNum := 333;
    -- 예외처리
    
    
    -- 예외처리
END;
/
 
/*이 주석은 사용 가능*/
-- 호출
VAR val NUMBER
-- 프로시저 호출
EXECUTE myproc(111, :val);
--EXEC
 
PRINT val;
-- 매개변수가 없을때는 아에 ()도 넣지 않을것
CREATE OR REPLACE PROCEDURE hellProc 
IS
    msg VARCHAR2(10);
BEGIN
    msg := 'hellProc';
    DBMS_OUTPUT.PUT_LINE(msg || '호출');
END;
/
 
EXEC hellproc;
 
-- departments row추가
INSERT into departments(department_id, department_name, location_id)
VALUES(300'관리부'1400);
 
ROLLBACK;
 
CREATE OR REPLACE PROCEDURE add_dept(p_deptno IN departments.department_id%TYPE,
                                    p_deptname IN departments.department_name%TYPE,
                                    p_deptloc IN departments.location_id%TYPE)
 
IS
 
BEGIN
 
    INSERT into departments(department_id, department_name, location_id)
    VALUES(p_deptno, 'p_deptname', p_deptloc);
 
    --COMMIT;
EXCEPTION WHEN OTHERS THEN
    dbms_output.put_line('추가에 실패했습니다');
    -- ROLLBACK;
 
END;
 
/
 
EXECUTE add_dept(301'신규부서'2500);
 
 
-- update salary
-- 사원번호를 입력받아서 급여인상 30%
 
 
CREATE OR REPLACE PROCEDURE updateSal(v_empno IN NUMBER)
IS
BEGIN
    UPDATE employees
    SET salary = salary * 1.3
    WHERE employee_id = v_empno;
    
    -- COMMIT;
    EXCEPTION WHEN OTHERS THEN
    dbms_output.put_line('추가에 실패했습니다');
 
END;
/
 
ACCEPT empno PROMPT '사원번호:'
EXECUTE updatesal(&empno);
ROLLBACK;
 
-- 사원번호를 입력받으면 그 사원의 이름, 사원번호, 사원급여도 출력이 되도록 작성하라.
CREATE OR REPLACE PROCEDURE emp_info(p_empno IN employees.employee_id%TYPE)
IS
    v_emp employees%ROWTYPE;
BEGIN
    SELECT first_name, employee_id, salary
        INTO v_emp.first_name, v_emp.employee_id, v_emp.salary
    FROM employees
    where employee_id = p_empno;
    
    dbms_output.put_line('이름: '||v_emp.first_name);
    dbms_output.put_line('사원번호: '||v_emp.employee_id);
    dbms_output.put_line('급여: '||v_emp.salary);
END;
/
 
EXECUTE emp_info(102);
 
/*
    Cursor : 저장 주소 공간 -> pointer
    암시적 커서 : 자동 생성
    
        SQL%ROWCOUNT : ROW의 수, 기본 제공(LENGTH와 유사하다)
        SQL%FOUND : ROW의 수가 한개이상일 경우
        SQL%NOTFOUND : ROW의 수가 0
          
    명시적 커서 : 수동 생성
*/
-- 암시적 커서
CREATE OR REPLACE PROCEDURE implicit_cursor(p_empname IN employees.first_name%TYPE)
IS
    v_sal employees.salary%TYPE;
    v_update_row NUMBER;    -- 몇개의 행이 수정되는지 조사하는 부분의 변수 
BEGIN
    -- 검색
    SELECT salary into v_sal
    FROM employees
    where first_name = p_empname;
    
    -- 검색된 데이터가 있는 경우
    IF SQL%FOUND THEN
        DBMS_OUTPUT.PUT_LINE('검색된 데이터가 있습니다.');
    END IF;
    
    -- 수정
    UPDATE employees
    SET salary = salary * 1.1
    where first_name = p_empname;
    
    -- 출력
    -- 지금 업데이트 된 사원의 숫자 
    v_update_row := SQL%ROWCOUNT;
    DBMS_OUTPUT.PUT_LINE('급여가 인상된 사원 수' || v_update_row);
    
EXCEPTION WHEN NO_DATA_FOUND THEN
    DBMS_OUTPUT.PUT_LINE('검색 된 데이터가 없습니다.');
    
END;
/
 
EXECUTE implicit_cursor('Guy');
 
ROLLBACK;
 
-- 명시적 커서
CREATE OR REPLACE PROCEDURE expCursor_test(v_deptno IN departments.department_id%TYPE)
IS
    CURSOR dept_avg
    IS
    SELECT d.department_name, COUNT(e.employee_id) CNT,
        ROUND(AVG(salary), 3) SAL
    FROM employees e, departments d
    WHERE e.department_id = d.department_id
        AND e.department_id = v_deptno
    GROUP BY d.department_name;
    
    -- CURSOR에 PATCH하기 위한 변수들을 선언
    v_dname departments.department_name%TYPE;
    emp_cnt NUMBER;
    sal_avg NUMBER;   
    
BEGIN
    -- CURSOR OPEN    
    OPEN dept_avg;
    
    -- CURSOR FETCH
    FETCH dept_avg INTO v_dname, emp_cnt, sal_avg;
 
    DBMS_OUTPUT.PUT_LINE('부서명:' || v_dname);
    DBMS_OUTPUT.PUT_LINE('사원수:' || emp_cnt);
    DBMS_OUTPUT.PUT_LINE('급여평균:' || sal_avg);
 
    -- CURSOR CLOSE
    CLOSE dept_avg;
END;
/
 
EXECUTE expcursor_test(50);
cs

'Oracle > PL' 카테고리의 다른 글

PL Trigger  (0) 2019.12.23
PL 응용문제 (프로시져, 함수)  (0) 2019.12.21
PL function(함수)  (0) 2019.12.20
PL 기초 활용  (0) 2019.12.19