bbossgroups 開發系列文章之-最佳實踐
一、概述
所謂最佳實踐,就是将采用bbossgroups架構體系開發業務系統的一個最佳的實作方式介紹給大家,本最佳實踐包含以下内容:

本文基本功能點如下:
1.bboss mvc架構基礎配置
2.bboss mvc控制器配置檔案
3.資料庫通路元件及sql語句配置檔案
4.dao元件管理及注入資料庫通路元件
5.業務元件管理及注入dao元件
6.業務開發前台和背景銜接的粘合劑-mvc控制器,注入業務元件
7.附帶介紹一下jsp頁面的内容
8.bboss mvc請求分派器及url映射配置
9.bboss mvc參數編碼過濾器配置
下面詳細介紹各個部分。
1.bboss mvc架構基礎配置
bboss mvc架構基礎配置,主要介紹bboss-mvc.xml檔案的配置内容,bboss-mvc.xml是bboss mvc架構的系統配置檔案,一般開發人員無需關心。詳細介紹請參考部落格文章:
《bboss mvc基礎配置介紹》
http://yin-bp.iteye.com/blog/11396082.bboss mvc控制器配置檔案
本節介紹mvc架構的一個典型的控制器配置檔案的内容:
<properties>
<property name="/uddi/requester/*.page"
path:main="/uddi/requester/main.jsp"
path:add-requester="/uddi/requester/addRequester.jsp"
path:edit-requester="/uddi/requester/editRequester.jsp"
path:query-requester="/uddi/requester/RequesterListInfo.jsp"
class="com.chinacreator.esb.uddi.requester.web.RequesterController"
f:fuzzySearch="true" f:requesterService="attr:uddi.requester.requesterService"
f:businessLineService="attr:businessLineService"/>
<property name="uddi.requester.requesterService"
class="com.chinacreator.esb.uddi.requester.service.impl.RequesterServiceImpl" f:requesterDao="attr:uddi.requester.requesterDao"/>
<property name="uddi.requester.requesterDao"
class="com.chinacreator.esb.uddi.requester.dao.impl.RequesterDaoImpl" f:executor="attr:uddi.requester.Configexecutor"/>
<property name="uddi.requester.Configexecutor"
class="com.frameworkset.common.poolman.ConfigSQLExecutor">
<construction>
<property value="com/chinacreator/esb/uddi/requester/dao/impl/sqlfile.xml"/>
</construction>
</property>
</properties>
從檔案内容可以看出控制器配置檔案中包含四大部分内容:
1.控制器配置(url映射規則,跳轉路徑配置,依賴的業務元件配置)
2.業務元件配置(元件及元件依賴的dao元件的配置)
3.dao元件的配置(dao元件及持久層操作對象配置)
4、持久層元件配置(持久層元件及元件引用的sql檔案路徑配置)
這四部分内容基本上郎闊了當今企業資訊系統開發的各個方面,表示層,控制層,業務層,持久層。從這裡也可以看出bbossgroups為企業j2ee項目架構提供了完整的解決方案,它通過aop/ioc、mvc将各個層面很好地編織起來,下面的章節會逐個介紹每部分,我們從持久層開始介紹。
3.資料庫通路元件及sql語句配置檔案
資料庫通路元件及注入dao元件,下面的配置聲明了一個com.frameworkset.common.poolman.ConfigSQLExecutor元件名字叫uddi.requester.Configexecutor:
<property name="uddi.requester.Configexecutor"
class="com.frameworkset.common.poolman.ConfigSQLExecutor">
<construction>
<property value="com/chinacreator/esb/uddi/requester/dao/impl/sqlfile.xml"/>
</construction>
</property>
dao元件通過名稱uddi.requester.Configexecutor來注入和引用該元件。這個持久元件配置了一個sql檔案:
com/chinacreator/esb/uddi/requester/dao/impl/sqlfile.xml
非常明顯我們将sql語句配置在了一個xml檔案中,我們來看看這個檔案的内容,其實配置檔案中我們用key和value的方式管理sql語句,我們在dao中通過key來引用每個sql語句。在開發環境可以将這個sql檔案配置為熱加載,即修改後不用重新開機應用伺服器就生效,sqlfile.xml檔案内容如下:
<?xml version="1.0" encoding="UTF-8"?>
<properties>
<property name="getRequesterDaoListInfo">
<![CDATA[
select r.*, o.ORG_NAME as service_requester_org_name from TD_ESB_REQUESTER r left join TD_SM_ORGANIZATION o
on r.SERVICE_REQUESTER_ORG = o.ORG_ID
where 1=1
#if($service_requester_code && !$service_requester_code.equals(""))
and SERVICE_REQUESTER_CODE = #[service_requester_code]
#end
#if($service_requester_account && !$service_requester_account.equals(""))
and SERVICE_REQUESTER_ACCOUNT like #[service_requester_account]
#end
#if($service_requester_name && !$service_requester_name.equals(""))
and SERVICE_REQUESTER_NAME like #[service_requester_name]
#end
#if($service_requester_org && !$service_requester_org.equals(""))
and SERVICE_REQUESTER_ORG = #[service_requester_org]
#end
order by create_time desc
]]>
</property>
<property name="delete">
<![CDATA[
delete from TD_ESB_REQUESTER where SERVICE_REQUESTER_ID = ?
]]>
</property>
<property name="findObjectById">
<![CDATA[
select r.*, o.ORG_NAME as service_requester_org_name from TD_ESB_REQUESTER r left join TD_SM_ORGANIZATION o
on r.SERVICE_REQUESTER_ORG = o.ORG_ID where SERVICE_REQUESTER_ID = ?
]]>
</property>
<property name="insert">
<![CDATA[
insert into TD_ESB_REQUESTER(ADDRESS ,BUSINESS_LINE_CODE ,CONTACT ,CREATE_TIME ,CREATOR ,EMAIL ,MODIFIER ,MODIFY_TIME ,PHONE ,SERVICE_REQUESTER_ACCOUNT ,SERVICE_REQUESTER_DESP ,SERVICE_REQUESTER_ID ,SERVICE_REQUESTER_NAME ,SERVICE_REQUESTER_ORG ,SERVICE_REQUESTER_PASSWORD ,USED_FLAG) values(#[address] ,#[business_line_code] ,#[contact] ,#[create_time] ,#[creator] ,#[email] ,#[modifier] ,#[modify_time] ,#[phone] ,#[service_requester_account] ,#[service_requester_desp] ,#[service_requester_id] ,#[service_requester_name] ,#[service_requester_org] ,#[service_requester_password] ,#[used_flag])
]]>
</property>
<property name="update">
<![CDATA[
update TD_ESB_REQUESTER set ADDRESS=#[address] ,BUSINESS_LINE_CODE=#[business_line_code] ,CONTACT=#[contact] ,EMAIL=#[email] ,MODIFIER=#[modifier] ,MODIFY_TIME=#[modify_time] ,PHONE=#[phone] ,SERVICE_REQUESTER_ACCOUNT=#[service_requester_account] ,SERVICE_REQUESTER_DESP=#[service_requester_desp] ,SERVICE_REQUESTER_ID=#[service_requester_id] ,SERVICE_REQUESTER_NAME=#[service_requester_name] ,SERVICE_REQUESTER_ORG=#[service_requester_org] ,SERVICE_REQUESTER_PASSWORD=#[service_requester_password] where SERVICE_REQUESTER_ID = #[service_requester_id]
]]>
</property>
<property name="updateFlag">
<![CDATA[
update TD_ESB_REQUESTER set USED_FLAG = #[used_flag], Modifier = #[modifier], Modify_time = #[modify_time]
where SERVICE_REQUESTER_ID = #[service_requester_id]
]]>
</property>
</properties>
另外檔案中的sql語句包含了模闆sql(帶#[service_requester_id]形态變量的sql語句)、動态sql(帶邏輯判斷的sql,#if()#end),預編譯sql(帶?号占位符的sql),最終持久層全部采用預編譯方式執行這些sql語句。
需要說明的是,bbossgroups 3.x 中sql檔案的重新整理機制配置在bboss-aop.jar的aop.properties檔案中,為<0時,将屏蔽重新整理功能:
sqlfile.refresh_interval=5000
dao元件管理及注入資料庫通路元件 本節内容介紹dao元件的配置和代碼,以及其如何通過注入方式引用上節中配置的持久層元件
uddi.requester.Configexecutor
首先看配置:
<property name="uddi.requester.requesterDao"
class="com.chinacreator.esb.uddi.requester.dao.impl.RequesterDaoImpl" f:executor="attr:uddi.requester.Configexecutor"/>
配置非常簡單,指定了dao元件的名字uddi.requester.requesterDao(我們将通過這個名稱将dao元件注入到業務元件中),實作類
com.chinacreator.esb.uddi.requester.dao.impl.RequesterDaoImpl
以及指定了屬性executor的值attr:uddi.requester.Configexecutor,這是一個引用,引用uddi.requester.Configexecutor對應的資料庫元件。
然後我們看看dao元件的實作類,從中我們可以看出dao中的方法是怎樣通過ConfigSQLExecutor來操作通路資料庫的:
package com.chinacreator.esb.uddi.requester.dao.impl;
import java.util.UUID;
import com.chinacreator.esb.datareuse.util.Constants;
import com.chinacreator.esb.uddi.requester.dao.RequesterDao;
import com.chinacreator.esb.uddi.requester.entity.Requester;
import com.frameworkset.common.poolman.ConfigSQLExecutor;
import com.frameworkset.util.ListInfo;
public class RequesterDaoImpl implements RequesterDao {
private static String dbName = Constants.DATAREUSE_DBNAME;
private ConfigSQLExecutor executor;
public ConfigSQLExecutor getExecutor() {
return executor;
}
public void setExecutor(ConfigSQLExecutor executor) {
this.executor = executor;
}
public ListInfo getRequesterDaoListInfo(String sortKey, boolean desc,
long offset, int pagesize, Requester queryCondObj) throws Exception {
// TODO Auto-generated method stub
return executor.queryListInfoBeanWithDBName(Requester.class,
dbName, "getRequesterDaoListInfo", offset, pagesize, queryCondObj);
}
public void delete(String id) throws Exception {
// TODO Auto-generated method stub
String delId = (String)id;
executor.deleteByKeysWithDBName(dbName, "delete", delId);
}
public Requester findObjectById(String id) throws Exception {
// TODO Auto-generated method stub
String findId = (String)id;
Requester requester = executor.queryObjectWithDBName(Requester.class, dbName, "findObjectById", findId);
return requester;
}
public void insert(Requester obj) throws Exception {
// TODO Auto-generated method stub
if(obj.getService_requester_id()== null||obj.getService_requester_id().equals("")){
obj.setService_requester_id(UUID.randomUUID().toString());}
executor.insertBean(dbName, "insert", obj);
}
public void update(Requester obj) throws Exception {
// TODO Auto-generated method stub
executor.updateBean(dbName, "update", obj);
}
public void updateFlag(Requester obj) throws Exception {
// TODO Auto-generated method stub
executor.updateBean(dbName, "updateFlag", obj);
}
}
其中我們可以看到dao在利用executor執行資料庫增删改查操作時,每個操作的參數分為3部分,一部分時dbName,指定操作針對poolman.xml檔案中配置的資料源datasource的名稱dbname的值,第二部分就是我們在sql檔案中配置的sql語句對應的名稱,第三部分就是相應操作需要的業務參數。其實還有可選的一部分就是分頁參數,例如
return executor.queryListInfoBeanWithDBName(Requester.class,
dbName, "getRequesterDaoListInfo", offset, pagesize, queryCondObj);
中的offset,pagesize兩個分别代表了分頁查詢的記錄起始位置和每頁最多傳回的記錄數。
補充一下dbname的在資料源配置檔案poolman.xml檔案(該檔案一般部署于classes目錄下)中是怎麼配置的:
<?xml version="1.0" encoding="gb2312"?>
<poolman>
<datasource>
<dbname>bspf</dbname>
<loadmetadata>false</loadmetadata>
<jndiName>jdbc/mysql-ds</jndiName>
<driver>oracle.jdbc.driver.OracleDriver</driver>
<url>jdbc:oracle:thin:@172.16.25.219:1521/orcl</url>
<username>esb</username>
<password>esb</password>
.........
</datasource>
<datasource external="true">
<dbname>datareuse</dbname>
<externaljndiName>jdbc/mysql-ds</externaljndiName>
<showsql>true</showsql>
</datasource>
</poolman>
更加詳細的配置請參考文章《
bbossgroups持久層架構資料源配置檔案執行個體》
業務元件管理及注入dao元件,本節講解業務元件配置部分的内容:
<property name="uddi.requester.requesterService"
class="com.chinacreator.esb.uddi.requester.service.impl.RequesterServiceImpl" f:requesterDao="attr:uddi.requester.requesterDao"/
業務元件配置和dao元件配置差不多,首先是業務元件的名字uddi.requester.requesterService(該名稱用來作為控制器引用業務元件的名稱),元件的實作類com.chinacreator.esb.uddi.requester.service.impl.RequesterServiceImpl,該業務元件引用的dao元件requesterDao,attr:uddi.requester.requesterDao,這裡引用的就是上節中介紹的dao元件,我們可以看看業務元件是怎麼通過dao來完成具體的業務操作的:
package com.chinacreator.esb.uddi.requester.service.impl;
import com.chinacreator.esb.uddi.requester.dao.RequesterDao;
import com.chinacreator.esb.uddi.requester.entity.Requester;
import com.chinacreator.esb.uddi.requester.service.RequesterService;
import com.frameworkset.util.ListInfo;
public class RequesterServiceImpl implements RequesterService {
private RequesterDao requesterDao;
public RequesterDao getRequesterDao() {
return requesterDao;
}
public void setRequesterDao(RequesterDao requesterDao) {
this.requesterDao = requesterDao;
}
public void insertRequester(Requester obj) throws Exception {
// TODO Auto-generated method stub
requesterDao.insert(obj);
}
public void updateRequester(Requester obj) throws Exception {
// TODO Auto-generated method stub
requesterDao.update(obj);
}
public void updateRequesterFlag(Requester obj) throws Exception {
// TODO Auto-generated method stub
requesterDao.updateFlag(obj);
}
public void deleteRequester(String id) throws Exception {
// TODO Auto-generated method stub
requesterDao.delete(id);
}
public Requester findRequesterById(String id) throws Exception {
// TODO Auto-generated method stub
return requesterDao.findObjectById(id);
}
public ListInfo getRequseterListInfo(String sortKey, boolean desc,
long offset, int pagesize, Requester req) throws Exception {
// TODO Auto-generated method stub
return requesterDao.getRequesterDaoListInfo(sortKey, desc, offset, pagesize, req);
}
}
業務元件的内容再簡單不過,這裡無需多說,我們直接看下節mvc如何使用業務元件來完成前端請求處理,并調整到結果處理頁面的。
業務開發前台和背景銜接的粘合劑-mvc控制器 ,介紹控制器是如何承前啟後地完成一個完整的業務處理請求的。
首先我們來看看mvc控制的的配置部分:
<property name="/uddi/requester/*.page"
path:main="/uddi/requester/main.jsp"
path:add-requester="/uddi/requester/addRequester.jsp"
path:edit-requester="/uddi/requester/editRequester.jsp"
path:query-requester="/uddi/requester/RequesterListInfo.jsp"
class="com.chinacreator.esb.uddi.requester.web.RequesterController"
f:fuzzySearch="true" f:requesterService="attr:uddi.requester.requesterService"
f:businessLineService="attr:businessLineService"/>
這個配置非常簡潔(bbossgroups aop架構配置簡潔的優良特質得以充分展現),也非常直覺,配置可以大緻分為以下部分:
1.控制url映射規則/uddi/requester/*.page,這個規則可以讓前端請求精确比對到具體的控制器,*代表将控制器實作類中的所有業務方法開放接收前台送出請求。
2.控制方法跳轉位址配置
path:main="/uddi/requester/main.jsp"
path:add-requester="/uddi/requester/addRequester.jsp"
path:edit-requester="/uddi/requester/editRequester.jsp"
path:query-requester="/uddi/requester/RequesterListInfo.jsp"
我們把控制器節點上的path:字首的屬性當成是跳轉頁面位址配置,而把f:開頭的屬性作為控制元件的屬性注入參數來處理
3.控制器類配置
class="com.chinacreator.esb.uddi.requester.web.RequesterController"
4.需要注入到控制器的業務元件和其他屬性配置:
f:fuzzySearch="true" f:requesterService="attr:uddi.requester.requesterService"
f:businessLineService="attr:businessLineService"
好,我們來具體看看控制器類得代碼吧(辛苦了,呵呵),控制器類得實作也很簡單:
package com.chinacreator.esb.uddi.requester.web;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.sql.Timestamp;
import java.util.Date;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.codehaus.jackson.map.ObjectMapper;
import org.frameworkset.util.annotations.PagerParam;
import org.frameworkset.util.annotations.RequestParam;
import org.frameworkset.web.servlet.ModelAndView;
import org.frameworkset.web.servlet.ModelMap;
import com.chinacreator.esb.AjaxResponseBean;
import com.chinacreator.esb.DropListEntity;
import com.chinacreator.esb.datareuse.businessline.service.BusinessLineService;
import com.chinacreator.esb.datareuse.util.Constants;
import com.chinacreator.esb.tools.StringTool;
import com.chinacreator.esb.uddi.requester.entity.Requester;
import com.chinacreator.esb.uddi.requester.service.RequesterService;
import com.chinacreator.security.AccessControl;
import com.frameworkset.util.ListInfo;
public class RequesterController {
private RequesterService requesterService;
private BusinessLineService businessLineService;
public BusinessLineService getBusinessLineService() {
return businessLineService;
}
public void setBusinessLineService(BusinessLineService businessLineService) {
this.businessLineService = businessLineService;
}
public RequesterService getRequesterService() {
return requesterService;
}
public void setRequesterService(RequesterService requesterService) {
this.requesterService = requesterService;
}
private boolean fuzzySearch;
public boolean isFuzzySearch() {
return fuzzySearch;
}
public void setFuzzySearch(boolean fuzzySearch) {
this.fuzzySearch = fuzzySearch;
}
/*
* 進入首頁面
*/
public ModelAndView main(){
ModelAndView view = new ModelAndView("path:main");
return view;
}
/**
* 将字元串轉成utf-8編碼
* @param str
* @return
* @throws UnsupportedEncodingException
*/
private String encode(String str) throws UnsupportedEncodingException {
return java.net.URLEncoder.encode(str, "utf-8");
}
/**
* 轉成json對象,直接用response的print方法
* @param response
* @param ajaxResponseBean
* @throws IOException
*/
private void write(HttpServletResponse response,
AjaxResponseBean ajaxResponseBean) throws IOException {
ObjectMapper objectMapper = new ObjectMapper();
String responseText = objectMapper.writeValueAsString(ajaxResponseBean);
response.getWriter().print(responseText);
}
/**
* 取得目前使用者的ID
* @param request
* @param response
* @return
*/
private String getUserId(HttpServletRequest request,
HttpServletResponse response) {
AccessControl accessControl = AccessControl.getInstance();
accessControl.checkAccess(request, response);
return accessControl.getUserID();
}
/*
* 查詢請求方資料
*/
public ModelAndView queryRequester(Requester requester,
@PagerParam(name = PagerParam.SORT)String sortKey,
@PagerParam(name = PagerParam.DESC)boolean desc,
@PagerParam(name = PagerParam.OFFSET)long offset,
@PagerParam(name = PagerParam.PAGE_SIZE)int pagesize,
ModelMap model){
String srAccount = requester.getService_requester_account();
String srName = requester.getService_requester_name();
if (isFuzzySearch()){
requester.setService_requester_account(StringTool.buildFuzzySearchString(srAccount, false));
requester.setService_requester_name(StringTool.buildFuzzySearchString(srName, false));
}
ListInfo listinfo = null;
try {
listinfo = requesterService.getRequseterListInfo(sortKey, desc, offset, pagesize, requester);
listinfo.setMaxPageItems(pagesize);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
ModelAndView view = new ModelAndView("path:query-requester", "listinfo", listinfo);
return view;
}
/*
* 進入新增請求方頁面
*/
public ModelAndView addRequester(Requester requster, ModelMap model) throws Exception {
addBusinessLineDropList(model);
ModelAndView view = new ModelAndView("path:add-requester");
return view;
}
public void createRequester(Requester requester,
@RequestParam(name = "service_requester_id") String service_requester_id,
ModelMap model, HttpServletRequest request,
HttpServletResponse response) throws Exception {
AjaxResponseBean ajaxResponseBean = new AjaxResponseBean();
String userId = this.getUserId(request, response);
try {
Date date = new Date();
Timestamp timestamp = new Timestamp(date.getTime());
requester.setCreator(userId);
requester.setCreate_time(timestamp);
requester.setModifier(userId);
requester.setModify_time(timestamp);
//賬号建立後預設為啟用
requester.setUsed_flag(Constants.UsedBoolean.TRUE.getValue());
requesterService.insertRequester(requester);
ajaxResponseBean.setStatus("success");
ajaxResponseBean.setData(requester.getService_requester_id());
} catch (Exception e) {
e.printStackTrace();
ajaxResponseBean.setStatus("error");
if (e.getMessage() != null && e.getMessage().indexOf("unique") > 0) {
ajaxResponseBean.setData(encode("賬号已存在!"));
} else {
throw e;
}
}
write(response, ajaxResponseBean);
}
/*
* 進入修改請求方頁面
*/
public ModelAndView editRequester(@RequestParam(name = "service_requester_id")String service_requester_id,
ModelMap model) throws Exception{
addBusinessLineDropList(model);
Requester obj = requesterService.findRequesterById(service_requester_id);
ModelAndView view = new ModelAndView("path:edit-requester", "obj", obj);
view.addObject("close", model.get("close"));
return view;
}
private void addBusinessLineDropList(ModelMap model) throws Exception{
List<DropListEntity> businessLines = businessLineService.getBusinessLineDropList();
model.addAttribute("businessLines", businessLines);
}
public void updateRequester(Requester requester,
@RequestParam(name = "service_requester_id") String service_requester_id,
ModelMap model, HttpServletRequest request,
HttpServletResponse response) throws Exception {
String userId = this.getUserId(request, response);
AjaxResponseBean ajaxResponseBean = new AjaxResponseBean();
try {
Date date = new Date();
Timestamp timestamp = new Timestamp(date.getTime());
requester.setModifier(userId);
requester.setModify_time(timestamp);
requesterService.updateRequester(requester);
ajaxResponseBean.setStatus("success");
} catch (Exception e) {
e.printStackTrace();
throw e;
}
write(response, ajaxResponseBean);
}
public void updateRequesterFlag(Requester requester,
HttpServletRequest request, HttpServletResponse response) throws Exception {
String userId = this.getUserId(request, response);
AjaxResponseBean ajaxResponseBean = new AjaxResponseBean();
try {
Date date = new Date();
Timestamp timestamp = new Timestamp(date.getTime());
requester.setModifier(userId);
requester.setModify_time(timestamp);
requesterService.updateRequesterFlag(requester);
ajaxResponseBean.setStatus("success");
} catch (Exception e) {
e.printStackTrace();
throw e;
}
write(response, ajaxResponseBean);
}
/*
* 啟用請求方
*/
public void startRequester(Requester requester,
HttpServletRequest request, HttpServletResponse response) throws Exception {
requester.setUsed_flag(Constants.UsedBoolean.TRUE.getValue());
updateRequesterFlag(requester, request, response);
}
/*
* 停用請求方
*/
public void stopRequester(Requester requester,
HttpServletRequest request, HttpServletResponse response) throws Exception {
requester.setUsed_flag(Constants.UsedBoolean.FALSE.getValue());
updateRequesterFlag(requester, request, response);
}
public void deleteRequester(
@RequestParam(name = "service_requester_id") String service_requester_id,
HttpServletResponse response) throws Exception {
AjaxResponseBean ajaxResponseBean = new AjaxResponseBean();
try {
requesterService.deleteRequester(service_requester_id);
ajaxResponseBean.setStatus("success");
} catch (Exception e) {
e.printStackTrace();
ajaxResponseBean.setStatus("error");
if (e.getMessage() != null && e.getMessage().indexOf("constraint") > 0) {
ajaxResponseBean.setData(encode("存在關聯資料,不能删除!"));
} else {
throw e;
}
}
write(response, ajaxResponseBean);
}
}
附帶介紹一下jsp頁面的内容 ,通過這個頁面内容我們可以看出視圖層是如何送出請求到控制器的,以及控制器處理完請求後怎麼把處理結果回報給視圖層的。
jsp頁面代碼片段,其它部分全部省略掉,js送出請求到控制 器,然後通過回調将響應添加到div的代碼片段:
$(document).ready(function(){
queryRequester();
}
function queryRequester(){
$('#queryForm').form('submit', {
"url": "queryRequester.page",
onSubmit:function(){
//顯示遮罩
blockUI();
},
success:function(responseText){
//去掉遮罩
unblockUI();
$("#RequesterListInfoContainer").html(responseText);
}
});
}
jsp代碼片段:
<div>
<form id="theForm" method="post">
<tr>
<td nowrap="nowrap" class="c2">
賬号:
</td>
<td width="30%">
<input class="input_default easyui-validatebox"
name="service_requester_account" type="text" validType="Caracters"
required="true" missingMessage="必填!" />
</td>
<td nowrap="nowrap" class="c2">
密碼:
</td>
<td width="30%">
<input class="input_default easyui-validatebox"
name="service_requester_password" type="password" validType="Caracters"
required="true" missingMessage="必填!" />
</td>
</tr>
</tbody>
<tfoot>
<tr>
<td align="center" colspan="4">
<button type="submit" class="button">
儲存
</button>
</td>
</tr>
</tfoot>
</table>
</form>
<div id="RequesterListInfoContainer">
</div>
呵呵,相信熟悉jquery的朋友對上述代碼并不陌生。
本節介紹bboss mvc請求分派器、url映射規則以及mvc配置檔案在web.xml檔案中的配置方法:
<servlet>
<servlet-name>mvc</servlet-name>
<servlet-class>org.frameworkset.web.servlet.DispatchServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/conf/uddi/bboss-*.xml,/WEB-INF/conf/datareuse/bboss-*.xml,/WEB-INF/conf/commons/bboss-*.xml,/WEB-INF/conf/bboss-*.xml</param-value>
</init-param>
<load-on-startup>0</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>mvc</servlet-name>
<url-pattern>*.page</url-pattern>
</servlet-mapping>
本節是本文的最後一節,介紹bboss mvc參數編碼過濾器在web.xml檔案中的配置:
<filter>
<filter-name>CharsetEncoding</filter-name>
<filter-class>com.frameworkset.common.filter.CharsetEncodingFilter</filter-class>
<init-param>
<param-name>RequestEncoding</param-name>
<param-value>GBK</param-value>
</init-param>
<init-param>
<param-name>ResponseEncoding</param-name>
<param-value>GBK</param-value>
</init-param>
<init-param>
<param-name>mode</param-name>
<param-value>0</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>CharsetEncoding</filter-name>
<url-pattern>*.jsp</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>CharsetEncoding</filter-name>
<url-pattern>*.do</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>CharsetEncoding</filter-name>
<url-pattern>*.frame</url-pattern>
</filter-mapping>
<filter-mapping>
<filter-name>CharsetEncoding</filter-name>
<url-pattern>*.page</url-pattern>
</filter-mapping>