摘要
本文以Oracle資料庫為例,介紹了在採用JSP技術開發WEB應用時一種簡單通用的表單資料儲存處理方法,以減輕開發工作量,同時提供了主要的程式碼。
引言
J2EE(Java 2 Enterprise Edition)技術已廣泛應用在Web應用開發中,其中的JavaBean、Servlet技術為開發者提供了更為清晰的開發環境,使用JSP技術表現頁面,使用Servlet技術完成大量的業務處理,使用Bean來儲存資料及一些業務處理。在WEB應用程式中,業務資料儲存到資料庫中的處理工作經常很繁重,其中一種主要的形式就是表單資料儲存到資料庫,整個應用程式處理過程牽涉到大量的這種資料儲存操作,對每個表單都要單獨編寫相應的資料儲存程序,花費了開發人員大量的時間和精力。採用什麼方法來減輕表單資料儲存的開發工作量是值得研究的問題。
兩種常見的表單資料儲存處理方法
1、對每一表單都編寫對應的程式碼
在JSP頁面或JavaBean或Servlet中,使用request. getparameter()函數逐一提取表單提交的數據,或編寫對應的JavaBean,使用setProperty方法將資料自動取到JavaBean中,然後產生SQL語句(insert,update,delete),最後執行executeupdate()函數完成資料表儲存。
2.對每一資料表自動產生一個JavaBean程式碼
資料庫系統必須支援使用者能夠讀取表格結構,並辨識關鍵欄位。利用物件導向快速開發工具,如PowerBuilder、Delphi等,自行開發一個java程式碼自動產生程式。在該程式中讀取資料庫表的結構:欄位名稱、資料型別、資料長度,自動產生一個JavaBean代碼。在這個程式碼中定義與表格中欄位對應的同名變量,建立所有變數的setValue和getValue方法,建立insert、update、delete函數分別處理insert、update、delete的SQL語句產生和執行。
在表單提交的資料處理頁面中,寫如下程式碼,將表單資料儲存到JavaBean:
<jsp:useBean id="table" class="table1_bean" />
<jsp:setProperty name="table" property="*" />
(註:table1_bean為上述自動產生的對應某一個表的JavaBean)
然後呼叫table1_bean中insert、update、delete函數完成資料表存儲,並傳回執行結果。如:
<%boolean success =table.insert(); %>
第一種方法簡單直觀,但對每一個表單都需要編寫對應的資料處理程序。對稍微大一點的應用,表單數量可能很多,開發工作量很大,開發工作效率低。表格結構變動如增加、減少欄位時,需修改對應的資料處理程序。
第二種方法相對第一種簡單得多,每一個資料表的資料處理由對應的JavaBean實現,JavaBean自動生成,不需寫,表結構變動時只需重新產生新的JavaBean,經過java編譯後覆寫原java類別即可。但此方法需要開發JavaBean自動產生程序,表結構變動時JavaBean需要重新產生和編譯。
介紹一種簡單通用的方法實作表單資料儲存
在WEB應用開發中,許多表單在經過前台瀏覽器端簡單的資料校驗後,提交後台伺服器,伺服器對資料不用作任何處理直接將資料儲存到一個資料表中。對這種情況,我們可以只寫一個程序,對這些表單統一處理,將資料儲存到對應的一個資料表中。此方法同樣要求資料庫系統支援表結構讀取和關鍵字段識別。我們採用JSP技術來寫該程序,程式檔案取名為DbdataStore.jsp。
1.呼叫格式
在網頁中表單的Action呼叫方法如下:
<Form Name=Frm1 Method=Post Action="DBdataStore.jsp? tablename=table1&OperType=…">
table1為資料將要儲存的資料庫表的表名,OperType操作類型分為三種:insert,update,delete。
表單中的<input type=text name=…>,<textarea name=…><select name=…>等中的name值應與資料表的欄位名稱相同,DBdataStore.jsp中逐一擷取表單提交的對應欄位名的資料值,若表單中未定義輸入,得到的值為空值,則對該欄位不處理。
2.以oracle為例的視圖定義
1) 建立表格各列資料型別檢視
CREATE OR REPLACE VIEW v_dbstru AS SELECT table_name,column_name,data_type,data_length,data_precision,data_scale,column_id
FROM all_tab_columns WHERE owner='user1';//user1為資料表的屬主。
2) 建立表格的關鍵列視圖
CREATE OR REPLACE VIEW v_pkey_column AS
SELECT b.table_name,b.column_name,b.position
FROM all_constraints a,all_cons_columns b
WHERE a.owner=b.owner AND a.constraint_name=b.constraint_name AND a.owner='user1' AND a.constraint_type='P';
3.主要程式碼
1) 程式初始化
String tablename=request.getParameter(" tablename");//提取表名
String OperType=request.getParameter("OperType");//擷取作業類型
String sFieldValue="";//存放表單提交的欄位資料值
String fieldname="",Datatype="" //存放欄位名,欄位資料類型
int iFieldvalue=0;
String updateSql="",whereSql=" where ",insSql1="",insSql2="",opSql="",strSql ="";
ResultSet rs1=null,rs2=null;
insSql1="insert into "+tablename+" (";
insSql2="values(";
2)產生sql語句關鍵字段部分
產生insert語句關鍵字段部分,如:insert into table1(id 和values(100));
只使用關鍵字段生成update,delete語句where部分,如:where id=100;
在操作類型為update時,網頁form表單中不會修改關鍵欄位的資料。
rs1=Stmt.executeQuery("SELECT column_name FROM v_pkey_column WHERE table_name='"+tablename+"'");//取關鍵字欄位名
while(rs1.next()){
fieldname=rs1.getString("column_name");
rs2=Stmt.executeQuery("SELECT data_type FROM v_dbstru WHERE table_name='"+tablename+"' AND column_name='"+fieldname+"'");//取關鍵欄位資料類型if(rs2.next()){
Datatype=rs2.getString("data_type");
sFieldValue=request.getParameter(fieldname.toLowerCase());
//產生insert語句關鍵字段部分if(OperType.equals("insert")){
insSql1+=fieldname+",";
if((sFieldValue==null) ){
//表單未提交關鍵欄位資料值時,本文只以數字型處理,資料值按下一流水號計算。
rs2= Stmt. executeQuery("SELECT max("+fieldname+")+1 FROM "+tablename);rs2. next();iFieldvalue=rs2.getInt(1);insSql2+=Integer.toString(iFieldvalue)+"," ; }else if(Datatype.equals("DATE")){
insSql2+= "To_Date('" + sFieldValue + "','YYYY-MM-DD'),";
}else if(Datatype.equals("VARCHAR2") || Datatype.equals("CHAR")){
insSql2+="'" + sFieldValue+"',";}
else /*NUMBER,FLOAT */ insSql2+=sFieldValue+",";}
//產生update,delete語句where部分:where fieldname=... AND
if(OperType.equals("update") || OperType.equals("delete")){
if(Datatype.equals("DATE")){
whereSql+=fieldname+"=To_Date('" + sFieldValue + "','YYYY-MM-DD') AND ";
}else if(Datatype.equals("VARCHAR2") || Datatype.equals("CHAR")){
whereSql+=fieldname+"='" + sFieldValue+"' AND ";}
else /*NUMBER,FLOAT */ whereSql+=fieldname+"="+ sFieldValue+" AND ";}
}
}
whereSql=whereSql.substring(0,whereSql.length()-4);
3)非關鍵字段部分sql語句產生
update語句,如:update table1 set column1=value1,… where id=100
insert語句,如:insert into table1(id,column1,…)values(100,value1,…)
updateSql="update "+tablename+" set ";
strSql="SELECT column_name,data_type,data_length,data_precision,data_scale FROM v_dbstru a "+"where table_name='"+tablename+"' AND a.column_name not in (SELECT b.column_tablename FROM_nameb. )";
rs1=Stmt.executeQuery(strSql);//取非關鍵字段字段名和資料類型
while(rs1.next()){
fieldname=rs1.getString("column_name");Datatype=rs1.getString("data_type");sFieldValue=request.getParameter(fieldname.toLowerCase());//若表單未提交該欄位的值,則忽略該欄位的值,則忽略該欄位的值,則忽略該欄位的值的處理if((sFieldValue!=null)){
//產生insert語句=insSql1+insSql2 即insert into tablename(… 和values(…
if(OperType.equals("insert")){ insSql1+=fieldname+",";
if(Datatype.equals("DATE")){
insSql2+= "To_Date('" + sFieldValue + "','YYYY-MM-DD'),";
} else if(Datatype.equals("VARCHAR2") || Datatype.equals("CHAR")){
insSql2+="'" + sFieldValue+"',";}else /*NUMBER,FLOAT*/ insSql2+= sFieldValue+",";}
//產生update語句=updateSql+whereSql 即update tablename set ... where fieldname=... if(OperType.equals("update")){
if(Datatype.equals("DATE")){
updateSql+=fieldname+"=To_Date('" + sFieldValue + "','YYYY-MM-DD'),";
}else if(Datatype.equals("VARCHAR2") || Datatype.equals("CHAR")){
updateSql+=fieldname+"='" + sFieldValue,1}+"',";}else /*NUMBER,FLOAT*/ updateSql+=fieldname+"="+sFieldValue+",";} ))
rs1.close();
4)產生完整的sql語句並執行
if(OperType.equals("insert"))
opSql=insSql1.substring(0,insSql1.length()-1)+")"+insSql2.substring(0,insSql2.length()-1)+")";
if(OperType.equals("update"))
opSql=updateSql.substring(0,updateSql.length()-1)+" "+whereSql;if(OperType.equals("delete"))
opSql="delete FROM "+tablename+" "+whereSql;
//已產生完整的sql語句opSql
try{sqlnrows=Stmt.executeUpdate(opSql);}
catch(SQLException e){out.println("SQLException:"+opSql);}
4、特點
該方法對所有這種直接儲存的表單都統一使用本程序,具有通用性,不必對每個表單或每個資料表獨立開發對應程序,開發工作量非常少,呼叫也非常簡單。同時,在表格結構變動時,不用修改DBdataStore.jsp程式。本程式也可改寫為Servelet,呼叫格式為<Form Name=Frm1 Method=Post Action="DBdataStoreServelet?tablename=table1&OperType=…">。
結論
在Web應用中,如果表單資料在提交後,還需要伺服器後台作進一步的資料校驗或處理,則需要採用第二種方法。但很多情況是使用者在表單中輸入或修改數據,在前台瀏覽器端使用javascript對資料進行簡單校驗或處理,然後提交,在後台伺服器端不作任何處理,直接將表單提交的資料儲存到資料庫的一個表中。這時候採用第三種方法非常簡便,可以大幅減輕開發人員的工作量。