JSP表單數(shù)據(jù)存儲(chǔ)的通用方法
J2EE(Java 2 Enterprise Edition)技術(shù)已廣泛應(yīng)用在Web應(yīng)用開(kāi)發(fā)中,其中的JavaBean、Servlet技術(shù)為開(kāi)發(fā)者提供了更為清晰的開(kāi)發(fā)環(huán)境,使用JSP技術(shù)表現(xiàn)頁(yè)面,使用Servlet技術(shù)完成大量的業(yè)務(wù)處理,使用Bean來(lái)存儲(chǔ)數(shù)據(jù)及一些業(yè)務(wù)處理。在WEB應(yīng)用中,業(yè)務(wù)數(shù)據(jù)存儲(chǔ)到數(shù)據(jù)庫(kù)中的處理工作經(jīng)常很繁重,其中一種主要的形式就是表單數(shù)據(jù)存儲(chǔ)到數(shù)據(jù)庫(kù),整個(gè)應(yīng)用處理過(guò)程牽涉到大量的這種數(shù)據(jù)存儲(chǔ)操作,對(duì)每個(gè)表單都要單獨(dú)編寫(xiě)相應(yīng)的數(shù)據(jù)存儲(chǔ)程序,花費(fèi)了開(kāi)發(fā)人員大量的時(shí)間和精力。采用什么方法來(lái)減輕表單數(shù)據(jù)存儲(chǔ)的開(kāi)發(fā)工作量是值得研究的問(wèn)題。
兩種常見(jiàn)的JSP表單數(shù)據(jù)存儲(chǔ)處理方法
1、對(duì)每一表單都編寫(xiě)相應(yīng)的程序代碼
在JSP頁(yè)面或JavaBean或Servlet中,使用request. getparameter()函數(shù)逐一提取表單提交的數(shù)據(jù),或編寫(xiě)相應(yīng)的JavaBean,使用setProperty方法將數(shù)據(jù)自動(dòng)取到JavaBean中,然后生成SQL語(yǔ)句(insert,update,delete),最后執(zhí)行executeupdate()函數(shù)完成數(shù)據(jù)表存儲(chǔ)。
2、對(duì)每一數(shù)據(jù)表自動(dòng)生成一個(gè)JavaBean程序代碼
數(shù)據(jù)庫(kù)系統(tǒng)必須支持用戶能夠讀取表結(jié)構(gòu),并識(shí)別關(guān)鍵字段。利用面向?qū)ο罂焖匍_(kāi)發(fā)工具,如PowerBuilder、Delphi等,自行開(kāi)發(fā)一個(gè)java代碼自動(dòng)生成程序。在該程序中讀取數(shù)據(jù)庫(kù)表的結(jié)構(gòu):字段名、數(shù)據(jù)類(lèi)型、數(shù)據(jù)長(zhǎng)度,自動(dòng)生成一個(gè)JavaBean代碼。在該代碼中定義與表中字段對(duì)應(yīng)的同名變量,建立所有變量的setValue和getValue方法,建立insert、update、delete函數(shù)分別處理insert、update、delete的SQL語(yǔ)句生成和執(zhí)行。
在表單提交的數(shù)據(jù)處理頁(yè)面中,編寫(xiě)如下代碼,將表單數(shù)據(jù)存儲(chǔ)到JavaBean中:
- <jsp:useBean id="table" class="table1_bean" />
- <jsp:setProperty name="table" property="*" />
(注:table1_bean為上述自動(dòng)生成的對(duì)應(yīng)某一個(gè)表的JavaBean)
然后調(diào)用table1_bean中insert、update、delete函數(shù)完成數(shù)據(jù)表存儲(chǔ),并返回執(zhí)行結(jié)果。如:
- <%boolean success =table.insert(); %>
JSP表單數(shù)據(jù)存儲(chǔ)的這兩種方法中,第一種方法簡(jiǎn)單直觀,但對(duì)每一表單都需要編寫(xiě)相應(yīng)的數(shù)據(jù)處理程序。對(duì)稍微大一點(diǎn)的應(yīng)用,表單數(shù)量可能很多,開(kāi)發(fā)工作量很大,開(kāi)發(fā)工作效率低。表結(jié)構(gòu)變動(dòng)如增加、減少字段時(shí),需修改相應(yīng)的數(shù)據(jù)處理程序。
第二種方法相對(duì)第一種簡(jiǎn)便得多,每一數(shù)據(jù)表的數(shù)據(jù)處理由對(duì)應(yīng)的JavaBean實(shí)現(xiàn),JavaBean自動(dòng)生成,不需編寫(xiě),表結(jié)構(gòu)變動(dòng)時(shí)只需重新生成新的JavaBean,經(jīng)java編譯后覆蓋原java類(lèi)即可。但該方法需要開(kāi)發(fā)JavaBean自動(dòng)生成程序,表結(jié)構(gòu)變動(dòng)時(shí)JavaBean需要重新生成和編譯。
介紹一種簡(jiǎn)便通用的方法實(shí)現(xiàn)JSP表單數(shù)據(jù)存儲(chǔ)
在WEB應(yīng)用開(kāi)發(fā)中,很多表單在經(jīng)過(guò)前臺(tái)瀏覽器端簡(jiǎn)單的數(shù)據(jù)校驗(yàn)后,提交后臺(tái)服務(wù)器,服務(wù)器對(duì)數(shù)據(jù)不用作任何處理直接將數(shù)據(jù)存儲(chǔ)到一個(gè)數(shù)據(jù)表中。對(duì)這種情況,我們可以只編寫(xiě)一個(gè)程序,對(duì)這些表單統(tǒng)一處理,將數(shù)據(jù)存儲(chǔ)到相應(yīng)的一個(gè)數(shù)據(jù)表中。該方法同樣要求數(shù)據(jù)庫(kù)系統(tǒng)支持表結(jié)構(gòu)讀取和關(guān)鍵字段識(shí)別。我們采用JSP技術(shù)編寫(xiě)該程序,程序文件取名為DbdataStore.jsp。
1、調(diào)用格式
在網(wǎng)頁(yè)中表單的Action調(diào)用方法如下:
- <Form Name=Frm1 Method=Post Action="DBdataStore.jsp? tablename=table1&OperType=…">
table1為數(shù)據(jù)將要存儲(chǔ)的數(shù)據(jù)庫(kù)表的表名,OperType操作類(lèi)型分為三種:insert,update,delete。
表單中的<input type=text name=…>,<textarea name=…><select name=…>等中的name值應(yīng)與數(shù)據(jù)表的字段名相同,DBdataStore.jsp中逐一提取表單提交的對(duì)應(yīng)字段名的數(shù)據(jù)值,若表單中未定義輸入,得到的值為空值,則對(duì)該字段不作處理。
2、以oracle為例的視圖定義
1) 建立表各列數(shù)據(jù)類(lèi)型視圖
- 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為數(shù)據(jù)表的屬主。
2) 建立表的關(guān)鍵列視圖
- 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");//提取操作類(lèi)型
- String sFieldValue="";//存放表單提交的字段數(shù)據(jù)值
- String fieldname="",Datatype="" //存放字段名,字段數(shù)據(jù)類(lèi)型
- int iFieldvalue=0;
- String updateSql="",whereSql=" where ",insSql1="",insSql2="",opSql="",strSql ="";
- ResultSet rs1=null,rs2=null;
- insSql1="insert into "+tablename+" (";
- insSql2="values(";
2)生成sql語(yǔ)句關(guān)鍵字段部分
生成insert語(yǔ)句關(guān)鍵字段部分,如:insert into table1(id 和 values(100));
只使用關(guān)鍵字段生成update,delete語(yǔ)句where部分,如:where id=100;
在操作類(lèi)型為update時(shí),網(wǎng)頁(yè)form表單中不對(duì)關(guān)鍵字段的數(shù)據(jù)進(jìn)行修改。
- rs1=Stmt.executeQuery("SELECT column_name FROM v_pkey_column WHERE table_name='"+tablename+"'");
- //取關(guān)鍵字段字段名
- 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+"'");//取關(guān)鍵字段數(shù)據(jù)類(lèi)型
- if(rs2.next()){
- Datatype=rs2.getString("data_type");
- sFieldValue=request.getParameter(fieldname.toLowerCase());
- //生成insert語(yǔ)句關(guān)鍵字段部分
- if(OperType.equals("insert")){
- insSql1+=fieldname+",";
- if((sFieldValue==null) ){
- //表單未提交關(guān)鍵字段數(shù)據(jù)值時(shí),本文只按數(shù)字型處理,數(shù)據(jù)值按下一流水號(hào)計(jì)算。
- 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,F(xiàn)LOAT */ insSql2+=sFieldValue+",";}
- //生成update,delete語(yǔ)句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,F(xiàn)LOAT */ whereSql+=fieldname+"="+ sFieldValue+" AND ";}
- }
- }
- whereSql=whereSql.substring(0,whereSql.length()-4);
3)非關(guān)鍵字段部分sql語(yǔ)句生成
update語(yǔ)句,如:update table1 set column1=value1,… where id=100
insert語(yǔ)句,如: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_name FROM v_pkey_column b where b.table_name=a.table_name)";
- rs1=Stmt.executeQuery(strSql);//取非關(guān)鍵字段字段名和數(shù)據(jù)類(lèi)型
- while(rs1.next()){
- fieldname=rs1.getString("column_name");Datatype=rs1.getString("data_type");
- sFieldValue=request.getParameter(fieldname.toLowerCase());
- //若表單未提交該字段的值,則忽略該字段的處理
- if((sFieldValue!=null)){
- //生成insert語(yǔ)句=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,F(xiàn)LOAT*/ insSql2+= sFieldValue+",";}
- //生成update語(yǔ)句=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,F(xiàn)LOAT*/
- updateSql+=fieldname+"="+sFieldValue+",";} ))
- rs1.close();
4)生成完整的sql語(yǔ)句并執(zhí)行
- 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語(yǔ)句opSql
- try{sqlnrows=Stmt.executeUpdate(opSql);}
- catch(SQLException e){out.println("SQLException:
- "+opSql);}
4、特點(diǎn)
該方法對(duì)所有這種直接存儲(chǔ)的表單都統(tǒng)一使用本程序,具有通用性,不必對(duì)每個(gè)表單或每個(gè)數(shù)據(jù)表獨(dú)立開(kāi)發(fā)相應(yīng)程序,開(kāi)發(fā)工作量非常少,調(diào)用也非常簡(jiǎn)便。同時(shí),在表結(jié)構(gòu)變動(dòng)時(shí),不用修改DBdataStore.jsp程序。本程序也可改寫(xiě)為Servelet,調(diào)用格式為<Form Name=Frm1 Method=Post Action="DBdataStoreServelet?tablename=table1&OperType=…">。
結(jié)束語(yǔ)
在Web應(yīng)用中,如果表單數(shù)據(jù)在提交后,還需要服務(wù)器后臺(tái)作進(jìn)一步的數(shù)據(jù)校驗(yàn)或處理,則需要采用第二種方法。但很多情況是用戶在表單中輸入或修改數(shù)據(jù),在前臺(tái)瀏覽器端使用javascript對(duì)數(shù)據(jù)進(jìn)行簡(jiǎn)單校驗(yàn)或處理,然后提交,在后臺(tái)服務(wù)器端不作任何處理,直接將表單提交的數(shù)據(jù)存儲(chǔ)到數(shù)據(jù)庫(kù)的一個(gè)表中。這時(shí)候采用第三種方法進(jìn)行JSP表單數(shù)據(jù)存儲(chǔ)非常簡(jiǎn)便,可以大大減輕開(kāi)發(fā)人員的工作量。
【編輯推薦】