明輝手游網(wǎng)中心:是一個(gè)免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺(tái)!

Oracle PL/SQL基礎(chǔ)之案例實(shí)戰(zhàn)

[摘要]前面已經(jīng)了解了關(guān)于PL/SQL編程的基礎(chǔ),本文將結(jié)合一個(gè)案例來(lái)加深對(duì)這些知識(shí)點(diǎn)的理解。一. 案例介紹某數(shù)據(jù)庫(kù)有兩張表,是關(guān)于某公司員工資料、薪水和部門信息的,它們分別是emp表和dept表,兩張表的...

  前面已經(jīng)了解了關(guān)于PL/SQL編程的基礎(chǔ),本文將結(jié)合一個(gè)案例來(lái)加深對(duì)這些知識(shí)點(diǎn)的理解。

  一. 案例介紹

  某數(shù)據(jù)庫(kù)有兩張表,是關(guān)于某公司員工資料、薪水和部門信息的,它們分別是emp表和dept表,兩張表的結(jié)構(gòu)如下:

  要求如下:

  1、按照上表結(jié)構(gòu)建立相應(yīng)的表,并每張表寫入5組合法數(shù)據(jù)。

  2、操縱相關(guān)表,使得"技術(shù)部"的員工的薪水上漲20%。

  3、建立日志,追蹤薪水變動(dòng)情況。

  4、建立測(cè)試包。

  二. 案例的分析與實(shí)現(xiàn)

  從前面案例的介紹不難看出,要求1考察點(diǎn)為基本SQL語(yǔ)句;要求2主要考察復(fù)合查詢;要求3是考察觸發(fā)器的應(yīng)用;要求4的考察面相對(duì)多一些,不僅考察了包的創(chuàng)建,而且也考察了在PL/SQL中的測(cè)試方法。了解了這些考察的知識(shí)點(diǎn),就可以一一去解決。

  要求1:

  首先根據(jù)前面表的結(jié)構(gòu)可以創(chuàng)建兩張表:

  ——創(chuàng)建員工表

create table emp (emp_id number(5), emp_name varchar2(20), emp_salary number(4));

  ——部門表

create table dept (dept_id number(3), dept_name varchar2(20), emp_id number(5));

  建立了表之后就可以往表里面寫數(shù)據(jù)了,這里把添加表記錄的代碼寫入到相應(yīng)的存儲(chǔ)過(guò)程。

/*給emp表添加記錄的存儲(chǔ)過(guò)程*/
create or replace procedure ins_table_emp(p_emp_id number,p_emp_name varchar2,p_emp_salary number) as
v_emp_id number:=p_emp_id;
v_emp_name varchar2(20):=p_emp_name;
v_emp_salary number:=p_emp_salary;
begin
 insert into emp values (v_emp_id,v_emp_name,v_emp_salary);
end ins_table_emp;

/*給dept表添加記錄的存儲(chǔ)過(guò)程*/
create or replace procedure ins_table_dept(p_dept_id number,p_dept_name varchar2,p_emp_id number) as
 v_dept_id number:=p_dept_id;
 v_dept_name varchar2(20):=p_dept_name;
 v_emp_id number:=p_emp_id;
begin
 insert into dept values (v_dept_id,v_dept_name,v_emp_id);
end ins_table_emp;

/*調(diào)用相應(yīng)的存儲(chǔ)過(guò)程實(shí)現(xiàn)記錄添加*/
begin
 ins_table_emp(10000,'',4000);
 ins_table_emp(10001,'??èy',2300);
 ins_table_emp(10002,'3?t',3500);
 ins_table_emp(10003,'à???',3500);
 ins_table_emp(10004,'á?ò?',3500);

 ins_table_dept(111,'DD?t2?',10000);
 ins_table_dept(111,'DD?t2?',10001);
 ins_table_dept(111,'DD?t2?',10002);
 ins_table_dept(112,'??ê?2?',10003);
 ins_table_dept(113,'êD3?2?',10004);
end;

  要求2:

  給指定部門的員工加薪,這實(shí)際上是一個(gè)復(fù)合查詢,首先需要把所有該部門的員工塞選出來(lái),然后對(duì)這些員工的薪水進(jìn)行相應(yīng)的改動(dòng)。依照這一思路,代碼如下:

 。ㄐ枰⒁獾氖牵簩⒁有降牟块T作為參數(shù),這樣的存儲(chǔ)過(guò)程更有靈活性。)

create or replace procedure add_salary(p_dept_name varchar2) as
v_dept_name varchar2(20):=p_dept_name;
begin
 update emp set emp.EMP_SALARY=emp.EMP_SALARY*1.2 where emp.EMP_ID in (select emp.EMP_ID from emp,dept where emp.EMP_ID=dept.EMP_ID and dept.DEPT_ID='??ê?2?');
end add_salary;

  要求3:

  建立日志對(duì)薪水的變動(dòng)情況形成一個(gè)追蹤,也就是說(shuō),如果對(duì)某個(gè)職員的薪水進(jìn)行變更就應(yīng)該將其相應(yīng)的變更記錄全部記下來(lái)。如果對(duì)emp表的salary字段創(chuàng)建一個(gè)觸發(fā)器,來(lái)監(jiān)視對(duì)salary的更改,把每次更改進(jìn)行記錄,這樣就達(dá)到了要求3的目的了。

create or replace trigger print_salary_change  
before delete or insert or update on emp  --觸發(fā)事件
for each row                  -- 每修改一行都需要調(diào)用此過(guò)程

declare           --只有觸發(fā)器的聲明需要declare,過(guò)程和函數(shù)都不需要
salary_balance number;
begin
--:new 與:old分別代表該行在修改前和修改后的記錄
salary_balance=:new.salary=:old.salary;
dbms_output.PUT_LINE('old salary is: ' :old.salary);
dbms_output.PUT_LINE('old salary is: ' :new.salary);
dbms_output.PUT_LINE('old salary is: ' to_char(salary_balance));
end print_salary_change;

  要求4:

  與其他語(yǔ)言(c/c++等)相比,PL/SQL的測(cè)試有其不同之處,歸納下來(lái)有三種方法:

  1、使用DBMS_OUTPUT包的PUT_LINE方法來(lái)顯示中間變量,以此來(lái)觀察程序是否存在邏輯錯(cuò)誤。

  2、插入測(cè)試表的方法。即創(chuàng)建一個(gè)臨時(shí)的中間表,然后把所有涉及到的中間變量的結(jié)果都作為記錄插入到中間表中,這樣可以查詢表中的結(jié)果來(lái)觀察程序的執(zhí)行情況。

  3、使用異常處理手段,對(duì)可疑的程序段使用begin … end ,然后可以在exception里進(jìn)行異常捕獲處理。

  這里準(zhǔn)備使用第二種方法來(lái)建立一個(gè)測(cè)試包,PL/SQL里包的概念類似于面向?qū)ο罄锏念惖母拍睿鼘⒁唤M操作和屬性封裝在一起,不僅增強(qiáng)了程序的模塊化,而且由于封裝了更多的操作和屬性而提高了執(zhí)行效能。建立一個(gè)PL/SQL需要兩個(gè)步驟:首先要建立包頭,類似于建立一個(gè)類的頭文件,里面主要對(duì)包中的過(guò)程,函數(shù)和變量的聲明;第二部分主要是包體部分,實(shí)現(xiàn)前面聲明的過(guò)程和函數(shù),另外還需要對(duì)包進(jìn)行初始化等工作。

  根據(jù)這一思路,建立測(cè)試包如下:

/*包頭部分*/
create or replace package debug as
procedure debug(v_description varchar2,v_valueOfvariable varchar2)
 procedure reset;
 v_numberOfLine number;
end debug;
/*包體部分*/
create or replace package body debug as
procedure debug(v_description varchar2,v_valueOfvariable varchar2) is
begin
 insert into debugtable
 values(v_numberOfLine,v_description,v_valueOfvariable);
 v_numberOfLine:=v_numberOfLine+1;
end debug;
procedure reset is
begin
 v_numberOfLine:=1;
 delete from debugtable;
end reset;
/*初始化部分*/
begin
 reset;
end debug;

  三.小結(jié)

  綜合前面對(duì)4個(gè)問(wèn)題的解答,基本把PL/SQL的主要部分融會(huì)進(jìn)來(lái)了,雖然很多地方只是涉及到比較粗淺的層次,但是有了這一基礎(chǔ),深入下去也是不難的。

  總之,PL/SQL編程與其他語(yǔ)言編程有一定的區(qū)別,讀者只有把握好其特點(diǎn)才能更好的掌握數(shù)據(jù)庫(kù)開(kāi)發(fā)的方面知識(shí)。

關(guān)鍵詞標(biāo)簽:實(shí)踐,案例,入門,varch

學(xué)習(xí)教程快速掌握從入門到精通的SQL知識(shí)。