《電子技術(shù)應(yīng)用》
您所在的位置:首頁 > 嵌入式技術(shù) > 業(yè)界動態(tài) > GKD-Base PL/SQL存儲函數(shù)實現(xiàn)的關(guān)鍵技術(shù)研究

GKD-Base PL/SQL存儲函數(shù)實現(xiàn)的關(guān)鍵技術(shù)研究

2008-10-24
作者:高朝瑞 熊偉 陳宏盛  翟玉人

  摘 要: 介紹具有自主知識產(chǎn)權(quán)的某安全數(shù)據(jù)庫管理系統(tǒng)" title="管理系統(tǒng)">管理系統(tǒng)GKD-Base" title="GKD-Base">GKD-Base的PL/SQL引擎,基于該引擎研究GKD-Base存儲函數(shù)機制實現(xiàn)的關(guān)鍵技術(shù)。設(shè)計了函數(shù)管理器" title="管理器">管理器和執(zhí)行狀態(tài)堆棧,通過語法樹表示存儲函數(shù)編譯后生成的中間代碼,并解決了中間代碼的執(zhí)行問題。最后實現(xiàn)了IN、OUT、INOUT三種參數(shù)模式函數(shù)的參數(shù)傳遞機制。
  關(guān)鍵詞: 存儲函數(shù) 過程式SQL語言 PL/SQL 函數(shù)管理器 執(zhí)行狀態(tài)堆棧


  GKD-Base是一個具有自主知識產(chǎn)權(quán)的安全數(shù)據(jù)庫管理系統(tǒng)。經(jīng)過十幾年的發(fā)展和試點應(yīng)用,已經(jīng)證明該系統(tǒng)是穩(wěn)定和安全可靠的。但是,隨著應(yīng)用需求向深度和廣度的發(fā)展,GKD-Base在很多方面暴露出不足,如不支持過程式SQL語言、不支持存儲過程和函數(shù)、缺少觸發(fā)器功能,這都為GKD-Base數(shù)據(jù)庫管理系統(tǒng)的進一步推廣應(yīng)用帶來了不小的障礙。由于目前數(shù)據(jù)庫管理系統(tǒng)產(chǎn)品大都支持過程式的SQL語言,如Oracle的PL/SQL、SQL Server的T-SQL、PostgreSQL的PL/pgSQL等,其中又以O(shè)racle的PL/SQL使用最為廣泛。因此,為使GKD-Base適應(yīng)應(yīng)用需求的發(fā)展,滿足數(shù)據(jù)庫管理和開發(fā)人員的需求,筆者開發(fā)了GKD-Base PL/SQL引擎,進而在引擎的基礎(chǔ)上實現(xiàn)了GKD-Base存儲函數(shù)機制。GKD-Base PL/SQL引擎兼容了Oracle PL/SQL V2.3語言規(guī)范,可以使用變量、類型、表達式、條件和循環(huán)、過程和函數(shù)等過程式語言中的通用機制進行程序設(shè)計,同時也可以使用SQL 語句進行數(shù)據(jù)操作。
  存儲函數(shù)是獨立存在于表之外的數(shù)據(jù)庫對象,是由用戶按照存儲函數(shù)語言規(guī)范編寫的經(jīng)過數(shù)據(jù)庫分析和編譯的PL/SQL程序,它有輸入/輸出參數(shù)和返回值,可以被其它應(yīng)用程序調(diào)用。存儲函數(shù)可以避免重復(fù)編碼,具有更高的可靠性和效率,是數(shù)據(jù)庫管理系統(tǒng)的重要功能擴充。本文在介紹GKD-Base PL/SQL引擎設(shè)計框架的基礎(chǔ)上,重點探討GKD-Base存儲函數(shù)實現(xiàn)的關(guān)鍵技術(shù)。
1 GKD-Base PL/SQL引擎的體系結(jié)構(gòu)
  GKD-Base采用客戶服務(wù)器結(jié)構(gòu),以多線程機制支持客戶訪問,登錄到數(shù)據(jù)庫的用戶實際上是運行數(shù)據(jù)庫管理系統(tǒng)映像的一個線程。GKD-Base支持TCP/IP協(xié)議,客戶可以在任何TCP/IP網(wǎng)絡(luò)和服務(wù)器建立連接訪問數(shù)據(jù)庫。GKD-Base API提供了用戶應(yīng)用、實用程序和第三方開發(fā)工具三種接入數(shù)據(jù)庫的方式,并提供了兩個級別的接口,即GKD-Base內(nèi)部接口GKD-API和GKD-Base ODBC Driver。GKD-API是用戶訪問數(shù)據(jù)庫的高效調(diào)用方式,用戶可利用GKD-API實現(xiàn)實時性要求較高的數(shù)據(jù)庫應(yīng)用,起到比較理想的效果。GKD-Base ODBC Driver是GKD-Base開放連接的手段,符合ODBC擴展一級。在GKD-Base的ODBC驅(qū)動程序支持下,用戶可利用各種第三方開發(fā)工具,方便地開發(fā)GKD-Base數(shù)據(jù)庫應(yīng)用程序。
  GKD-Base PL/SQL引擎應(yīng)用模式如圖1。PL/SQL語句塊通過PL/SQL編譯器編譯成語法樹形式表示的中間代碼,中間代碼信息可以保存在DBMS的中間代碼庫中??蛻魬?yīng)用程序中的請求由GKD-Base預(yù)處理器分離成為過程語句部分和SQL命令部分。過程語句解釋執(zhí)行器" title="執(zhí)行器">執(zhí)行器解釋執(zhí)行過程語句的中間代碼,SQL命令則直接調(diào)用GKD-API執(zhí)行。


  根據(jù)PL/SQL語言兼有過程式語句和SQL語句的特點,GKD-Base PL/SQL引擎把過程式語句和SQL語句分開處理。為了降低引擎實現(xiàn)的復(fù)雜程度,盡可能地從功能性的角度將整個引擎劃分為不同的處理模塊,每個處理模塊盡可能地降低相互間的耦合程度。其中最主要的劃分是將整個引擎劃分為前端編譯器和后端解釋執(zhí)行器。
  前端編譯器包括語言預(yù)處理、SQL語句分析、過程語句分析以及中間代碼生成等功能。PL/SQL語句塊輸入后,由語言預(yù)處理功能分離成SQL語句和過程語句。對于SQL語句,由GKD-Base SQL引擎解析后,建立SQL語句結(jié)點,進行相應(yīng)的變量綁定和語法檢查,檢查無誤后產(chǎn)生語法樹形式的中間代碼。對于過程語句,將語句成分進行語法分析,在分析中把聲明的變量和數(shù)據(jù)類型加入到符號表中,同時也產(chǎn)生語法樹形式的中間代碼。
  后端解釋執(zhí)行器的功能就是對前端編譯器生成的中間代碼進行解釋執(zhí)行。因此在結(jié)構(gòu)上,解釋執(zhí)行器與編譯器對應(yīng),也有過程語句執(zhí)行模塊和SQL語句執(zhí)行模塊。過程語句解釋執(zhí)行器解釋執(zhí)行過程語句的中間代碼,SQL命令則直接由SQL引擎執(zhí)行。另外,解釋器還包括例外處理模塊,負(fù)責(zé)錯誤檢查和報告。
  符號表是一個包含程序中的變量、自定義類型和函數(shù)信息的數(shù)據(jù)庫,它是GKD-Base PL/SQL引擎的核心數(shù)據(jù)結(jié)構(gòu)。它以一個關(guān)鍵字域(通常是一個符號的名字)為索引,一個關(guān)鍵字域的值對應(yīng)于庫中的一條記錄。每條記錄,即數(shù)據(jù)庫的一個項目,都對應(yīng)著一個符號的信息,如變量的類型或函數(shù)的返回值等。
2 GKD-Base PL/SQL過程與函數(shù)語言成分設(shè)計
  子程序" title="子程序">子程序把功能獨立并需要反復(fù)用到的代碼加以參數(shù)化處理,從而整合為一個命名模塊,在要使用這個模塊時就傳入具體參數(shù)值(如果需要)進行調(diào)用。與第三代語言中的子程序一樣,GKD-Base PL/SQL過程與函數(shù)是特定功能的邏輯抽象,由子程序定義和引用組成,可以分為過程和函數(shù)兩種類別。PL/SQL子程序可以分開編譯并存儲到數(shù)據(jù)庫中,用CREATE 語句建立,并成為模式對象的一部分。存儲過程/函數(shù)可以位于包結(jié)構(gòu)中,包分為包聲明和包體實現(xiàn)部分。本地過程/函數(shù)在DECLARE 節(jié)中的最后部分進行,而作為存儲過程和存儲函數(shù)將存儲在數(shù)據(jù)庫中,它們是命名的PL/SQL塊。這樣它們在結(jié)構(gòu)上也就沒有特別之處了。下面主要討論形參和實參的綁定設(shè)計。
  PL/SQL 函數(shù)中的參數(shù)說明帶有輸入和輸出描述,需要在編譯時刻進行處理,并對調(diào)用時的參數(shù)類型是否相容進行檢查。但在PL/SQL過程和函數(shù)的聲明中,限制CHAR、VARCHAR2參數(shù)的長度以及限制NUMBER的精度都是非法的。這是由于在調(diào)用一個函數(shù)時,實際參數(shù)的取值被傳遞進去,在函數(shù)內(nèi)部通過使用形式參數(shù)來引用這些實際參數(shù),不僅實際數(shù)值被傳遞進去,而且作為參數(shù)傳遞機制的一部分,對于這些變量的限制也被傳遞進去。例如:
  創(chuàng)建函數(shù):
  CREATE OR REPLACE FUNCTION ParameterLength (p_Var1 IN OUT VARCHAR2)
  RETURN VARCHAR2 AS
  BEGIN
    p_Var1:=′This is an example′;
    RETURN p_Var1; 
  END ParameterLength;
  調(diào)用函數(shù):
  DECLARE
  v_Variable1 VARCHAR (30);
  v_Variable2 VARCHAR (30);
  BEGIN
    v_Variable2:= ParameterLength (v_Variable1);
  END;
  上述調(diào)用將使p_Var1變量的最大長度為30(來自實參)。而如果將v_Variable1的長度定義成10,將產(chǎn)生錯誤。這個錯誤的產(chǎn)生不是過程設(shè)計不好,而是因為調(diào)用該過程的代碼發(fā)生了問題。值得注意的是:在過程定義中如果使用%TYPE作為形參類型的限制,而基準(zhǔn)類型又是受限的,那么該限制將作用于形式參數(shù)而不是實際參數(shù)。
  在缺省參數(shù)值的處理方式上,即當(dāng)調(diào)用時,沒有指明實際值時,就用缺省值。對使用缺省值的形式參數(shù),通常需要把它們放到參數(shù)表的末尾,這樣無論使用位置標(biāo)識法還是帶名標(biāo)識法都可以。
3 GKD-Base存儲函數(shù)實現(xiàn)的關(guān)鍵技術(shù)
3.1函數(shù)管理器

  GKD-Base PL/SQL引擎對PL/SQL程序的解釋執(zhí)行都通過一個內(nèi)部統(tǒng)一的入口,解釋器把語法樹載入到入口內(nèi)部的一個默認(rèn)的主控函數(shù),由這個函數(shù)驅(qū)動對其解釋執(zhí)行。這樣處理的主要目的是為了兼容子程序的語法分析和保持解釋器運轉(zhuǎn)機制的一致性。因為子程序就是PL/SQL程序中命名的語句塊,用戶調(diào)用子程序時,PL/SQL引擎可以再次調(diào)用內(nèi)部主控函數(shù)對其進行處理,并提交解釋器執(zhí)行。
  PL/SQL子程序包括局部聲明子程序和存儲子程序。為了管理和調(diào)用局部聲明子程序,設(shè)計了函數(shù)管理器,如圖2。函數(shù)管理器內(nèi)部定義了一系列數(shù)據(jù)結(jié)構(gòu)來描述局部聲明子程序調(diào)用的各種屬性,提供各種宏調(diào)用來獲取局部聲明子程序調(diào)用的參數(shù)和返回值,并提供必要的支持函數(shù)。從某種角度看,函數(shù)管理器也是用來存放各種局部聲明函數(shù)和過程信息的“符號表”,因為在管理器中還包括對聲明子程序聲明和調(diào)用信息的存儲、查找等功能。函數(shù)管理器提供一種結(jié)構(gòu)化的處理手段,以一種統(tǒng)一的方式實現(xiàn)對局部聲明子程序的調(diào)用。這種“統(tǒng)一”表現(xiàn)在:每個局部聲明子程序都在函數(shù)管理器中獲得一個識別號(Oid),這個識別號唯一標(biāo)識該局部聲明子程序;函數(shù)管理器維持一份記錄表格,其中存放所有局部聲明子程序的識別號、子程序名、子程序地址、參數(shù)個數(shù)和返回值類型等信息,同時接口還提供相應(yīng)的查找功能,源程序中的局部聲明子程序調(diào)用會被轉(zhuǎn)換成為子程序的識別號并通過該識別號查找得到子程序的詳細信息。


  所有局部聲明子程序都在函數(shù)管理器中進行注冊,存儲子程序并不在函數(shù)管理器中注冊,而是保存在數(shù)據(jù)庫中。對于一個子程序調(diào)用,首先根據(jù)子程序名到函數(shù)管理器中查找;如果沒有在函數(shù)管理器中找到再去查找數(shù)據(jù)庫,這樣就兼容了Oracle PL/SQL V2.3局部聲明子程序的定義優(yōu)先于存儲子程序的規(guī)則。
3.2 執(zhí)行狀態(tài)堆棧
  在GKD-Base PL/SQL引擎的解釋執(zhí)行器中,設(shè)計了一個 “執(zhí)行狀態(tài)”的數(shù)據(jù)結(jié)構(gòu),其中記載了存儲函數(shù)的執(zhí)行狀態(tài)信息,如函數(shù)中的聲明變量、函數(shù)的參數(shù)個數(shù)、函數(shù)的參數(shù)類型、函數(shù)每個參數(shù)的值、函數(shù)的返回值等。解釋器中所有的解釋函數(shù)都擁有一個指針參數(shù)指向這個結(jié)構(gòu),需要在這些函數(shù)間傳遞的信息(包括全局變量信息)將被拷貝到這個結(jié)構(gòu)??紤]到存儲函數(shù)是個單獨的PL/SQL語句塊層次,因此設(shè)計了執(zhí)行狀態(tài)堆棧。在調(diào)用存儲函數(shù)前,對當(dāng)前的現(xiàn)場進行保存,也就是將當(dāng)前的執(zhí)行狀態(tài)壓棧;在存儲函數(shù)調(diào)用結(jié)束之后恢復(fù)保存的現(xiàn)場,彈出當(dāng)前執(zhí)行狀態(tài),回到上一層執(zhí)行狀態(tài)繼續(xù)執(zhí)行,如圖3。


3.3 存儲函數(shù)的創(chuàng)建和調(diào)用
  用戶創(chuàng)建的存儲函數(shù)經(jīng)過編譯得到中間代碼并保存在GKD-Base的中間代碼庫中。函數(shù)調(diào)用時,根據(jù)函數(shù)名到GKD-Base的中間代碼庫中查找相應(yīng)存儲函數(shù)的中間代碼對其解釋執(zhí)行,最后得到返回值。GKD-Base存儲函數(shù)的創(chuàng)建和調(diào)用分別對應(yīng)中間代碼的產(chǎn)生和執(zhí)行過程。
  PL/SQL引擎對存儲函數(shù)的創(chuàng)建的具體處理過程如下:
 ?、倬幾g器得到用戶創(chuàng)建的存儲函數(shù);
 ?、趯τ脩魟?chuàng)建的存儲過程進行編譯,如果編譯通過則生成該存儲函數(shù)的語法樹;否則向用戶報告錯誤;
 ?、蹖⒋鎯瘮?shù)中間代碼保存在中間代碼庫中;存儲函數(shù)的源代碼保存在GKD-Base的系統(tǒng)表中,如表1;
 ?、芟蛴脩舴祷叵?。
  根據(jù)Oracle PL/SQL V2.3 語言規(guī)范,存儲函數(shù)調(diào)用本身不是一個語句,它只能作為其它語句的一部分。PL/SQL引擎對存儲函數(shù)的調(diào)用的具體過程如圖4:
  ①解釋器中語句鏈的解釋函數(shù)調(diào)用存儲函數(shù)的解釋函數(shù);
 ?、诮忉屍髟趫?zhí)行存儲函數(shù)之前,對當(dāng)前的現(xiàn)場進行保存,即將當(dāng)前的執(zhí)行狀態(tài)壓棧;
 ?、鄣綌?shù)據(jù)庫的中間代碼庫中找到存儲函數(shù)的語法樹,把存儲函數(shù)的語法樹掛在整個PL/SQL語句塊語法樹的過程調(diào)用節(jié)點上,并提交解釋器;
 ?、芙忉屍鲌?zhí)行存儲函數(shù)中間代碼;
 ?、萁忉屍髟诖鎯瘮?shù)調(diào)用結(jié)束后,將返回值返回給調(diào)用它的語句,并恢復(fù)保存的現(xiàn)場,彈出當(dāng)前執(zhí)行狀態(tài),回到上一層執(zhí)行狀態(tài)繼續(xù)執(zhí)行。


3.4 存儲函數(shù)的參數(shù)傳遞機制
  上述存儲函數(shù)調(diào)用過程中的一個主要難點就是實際參數(shù)和形式參數(shù)之間值的傳遞。與其它第三代語言一樣,用戶可以創(chuàng)建帶參數(shù)的PL/SQL存儲函數(shù)。這些參數(shù)可以通過值進行傳遞,也可以通過引用進行傳遞。
  GKD-Base PL/SQL引擎對存儲函數(shù)的形式參數(shù)的處理與對變量的處理是一致的,每個形式參數(shù)的屬性信息對應(yīng)著符號表中的一條記錄,形式參數(shù)的名字信息保存在相應(yīng)的名字堆棧層次內(nèi)。如本文第三部分所述,存儲函數(shù)的語法樹也是由解釋器載入到一個內(nèi)部默認(rèn)的主控函數(shù),由這個函數(shù)驅(qū)動對其解釋執(zhí)行。解釋器內(nèi)部的主控函數(shù)驅(qū)動存儲函數(shù)中所有變量的初始化,也驅(qū)動實際參數(shù)和形式參數(shù)之間值的傳遞。編譯器解析過程調(diào)用語句得到實際參數(shù)信息,首先判斷實際參數(shù)和形式參數(shù)的數(shù)據(jù)類型是否匹配,經(jīng)過類型檢查后,如果實際參數(shù)是常量或普通變量,直接將實際參數(shù)的值拷貝給形式參數(shù);如果實際參數(shù)是表達式,計算表達式的值傳給形式參數(shù);如果存儲函數(shù)的形式參數(shù)個數(shù)大于解析得到的實際參數(shù)個數(shù),二者之差為n,那么前面的形式參數(shù)取實際參數(shù)的值,后n個形式參數(shù)取語法樹中對應(yīng)的缺省值。
  在Oracle PL/SQL V2.3中,存儲函數(shù)的形式參數(shù)有三種模式:IN、OUT或INOUT。當(dāng)調(diào)用函數(shù)時,對IN參數(shù),實際參數(shù)的值傳遞給過程,函數(shù)結(jié)束后,實際參數(shù)沒有被改變;對OUT參數(shù),實際參數(shù)的任何值都被忽略,形式參數(shù)有一個NULL值,函數(shù)結(jié)束后,形式參數(shù)的內(nèi)容賦給實際參數(shù);對INOUT參數(shù),實際參數(shù)的值傳遞給過程,函數(shù)結(jié)束后,形式參數(shù)的內(nèi)容賦給實際參數(shù)。同時在存儲函數(shù)內(nèi)部,對IN參數(shù)賦值和把OUT參數(shù)賦給局部變量都是非法操作。
  在GKD-Base PL/SQL引擎中實現(xiàn)存儲函數(shù)三種形式參數(shù)模式的關(guān)鍵是執(zhí)行前后實際參數(shù)和形式參數(shù)的值的相互正確傳遞。在執(zhí)行存儲函數(shù)中間代碼前,判斷每個形式參數(shù)的模式,如果參數(shù)模式是IN或INOUT,直接把實際參數(shù)的值傳遞給形式參數(shù);如果模式是OUT,直接賦給形式參數(shù)一個初始的NULL值。對函數(shù)執(zhí)行體內(nèi)的賦值語句進行判斷檢查,如果賦值目標(biāo)是IN參數(shù)或把OUT參數(shù)賦給其它變量,都報告相應(yīng)的執(zhí)行錯誤。函數(shù)執(zhí)行結(jié)束后,解釋器彈出當(dāng)前執(zhí)行狀態(tài),回到上一層執(zhí)行狀態(tài)繼續(xù)執(zhí)行。對IN參數(shù),實際參數(shù)直接取其本身的值,即實際參數(shù)沒有改變;對OUT參數(shù)和INOUT參數(shù),根據(jù)執(zhí)行前形式參數(shù)和實際參數(shù)的對應(yīng)關(guān)系,用執(zhí)行完畢后的形式參數(shù)的值替代原來實際參數(shù)的值,即執(zhí)行完后形式參數(shù)的內(nèi)容賦給實際參數(shù)。
  PL/SQL 集成了一般過程式語言和說明性SQL 語言的特點,簡潔、高效,而其內(nèi)容卻十分豐富。這也從一個方面說明了作為過程式數(shù)據(jù)庫編程語言,在考慮到底層數(shù)據(jù)庫功能時所面臨的各種選擇的艱難。PL/SQL 語言的某些功能將隨著ORACLE 數(shù)據(jù)庫的發(fā)展而繼續(xù)增強,但是該語言的結(jié)構(gòu)是基本穩(wěn)定的。本文參照Oracle PL/SQL V2.3 語言規(guī)范,在某安全數(shù)據(jù)庫管理系統(tǒng)GKD-Base上開發(fā)了PL/SQL引擎作為該系統(tǒng)的重要擴充,并在該引擎基礎(chǔ)上實現(xiàn)了GKD-Base存儲函數(shù)機制,提供給用戶一種高效率的編程手段,增強了GKD-Base的功能。希望本文實現(xiàn)的PL/SQL引擎能對國產(chǎn)數(shù)據(jù)庫的開發(fā)起到借鑒作用。當(dāng)然PL/SQL 語言內(nèi)容十分豐富,本文不可能涉及到每一個設(shè)計細節(jié),PL/SQL 還在進一步發(fā)展,值得繼續(xù)關(guān)注。
  為了進一步擴展GKD-Base在安全保密領(lǐng)域的應(yīng)用前景,筆者將在對GKD-Base本身進一步完善的同時,繼續(xù)對GKD-Base PL/SQL引擎進行完善和改進。一方面在功能上實現(xiàn)封裝函數(shù)和過程的包;另一方面,在性能上對原有GKD-Base PL/SQL引擎進行優(yōu)化以提高編譯和執(zhí)行效率。
參考文獻
1 Ken Henderson. The Guru′s Guide to Transact-SQL. AddisonWesley PuB Co., 2000
2 PostgreSQL 7.4 Documentation.The PostgreSQL GloBal Development Group, 2003
3 Kenneth C.Louden著,馮博琴譯.編譯原理及實踐.北京:機械工業(yè)出版社,2000
4 Hector Garcia-Molina,Jeffrey D. Ullman,Jennifer Widom著.楊東青,唐世渭,徐其鈞譯. 數(shù)據(jù)庫系統(tǒng)實現(xiàn). 北京:機械工業(yè)出版社,2001
5 Levine,J.R.,lex與yacc(第二版).北京:機械工業(yè)出版社,2003

本站內(nèi)容除特別聲明的原創(chuàng)文章之外,轉(zhuǎn)載內(nèi)容只為傳遞更多信息,并不代表本網(wǎng)站贊同其觀點。轉(zhuǎn)載的所有的文章、圖片、音/視頻文件等資料的版權(quán)歸版權(quán)所有權(quán)人所有。本站采用的非本站原創(chuàng)文章及圖片等內(nèi)容無法一一聯(lián)系確認(rèn)版權(quán)者。如涉及作品內(nèi)容、版權(quán)和其它問題,請及時通過電子郵件或電話通知我們,以便迅速采取適當(dāng)措施,避免給雙方造成不必要的經(jīng)濟損失。聯(lián)系電話:010-82306118;郵箱:aet@chinaaet.com。