自己動手寫一個Struts2關鍵字: struts2, diy, 自己動手使用Struts2或webwork2有一段時間了,想把Struts2框架的思路簡單的與大家分享一下,之前我是看過Struts2源代碼的,所以本文算是它的一個功能非常有限的壓縮版本。我也不打算重復發(fā)明輪子,只想讓Struts2或Webwork2的新手更多的了解框架本身,而不僅僅是應用。廢話少說,開始吧。
本文采用基本Xml來配置Action,如果有時間會繼續(xù)寫Annotation的實現(xiàn)。Xml文件結構與Struts2的配置文件struts.xml幾乎一樣,這樣大家都比較熟悉,不過我簡寫了某些地方:
Struts.xml
<?xml version="1.0" encoding="UTF-8" ?> <!-- 為簡化框架,package的屬性都沒有實現(xiàn);而且所有的元素都是簡化版的 --> <struts> <package> <action name="hello" method="hello" class="com.leo.action.HelloAction"> <result name="success">/index.jsp</result> </action> </package> </struts>
是不是很熟悉啊,不過為了簡單我都給簡化了,否則這是一個沒有盡頭的工作。核心部分仍然是Filter,Struts2所有工作都是通過一個Filter來完成的(struts1.*是通過一個Action實現(xiàn)的)。我們先來看代碼:
public class StrutsFilter implements Filter {
public void doFilter(ServletRequest req, ServletResponse res,
FilterChain chain) throws IOException, ServletException {
// TODO Auto-generated method stub
HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) res;
ServletContext servletContext = filterConfig.getServletContext();
// 解析Request的URL和傳過來的參數(shù)
String actionName = StringUtil.parseServletPath(request
.getServletPath());
// 如果后綴不為.action,那么直接放過,不進行攔截
if (StringUtil.isEmpty(actionName)) {
chain.doFilter(request, response);
} else {
// 解析得到ActionClass,里面包括Action的類全名,返回頁面值,Action執(zhí)行的方法
ActionClass clas = this.getActionClass(actionName);
// 得到頁面的所有parameters參數(shù)(沒考慮上傳情況)
Map<String, String[]> params = request.getParameterMap();
// 為要調(diào)用的Action的set**方法設值,并返回要調(diào)用的Action對象本身
setBeforeActionValue(clas, params);
// 調(diào)用的Action執(zhí)行方法,并返回值設置在request域中
setResultValue(clas, request);
// 返回相應的JSP頁面
servletContext.getRequestDispatcher(clas.getResult()).forward(
request, response);
}
}
}
沒錯一些因果都因doFilter方法而起。我的作法是:
這就是大概的流程。因為是一個入門的框架所以很不完善,攔截器,result type的各種類型都沒有去實現(xiàn),因為我壓根沒想過要重復發(fā)明輪子。好,我們開始一步一步的看。
我們先來看這一句:ActionClass clas = this.getActionClass(actionName);
public class ActionClass {
/**
* 類名
*/
private String className;
/**
* 要調(diào)用的方法名
*/
private String method;
/**
* 返回結果頁面
*/
private String result;
/**
* 臨時存儲Action下的所有result結點
*/
private List<Element> elements = new ArrayList<Element>();
/**
* 要調(diào)用的Action本身
*/
private Object action;
//省略所有的set,get方法
}
ActionClass主要是用來存放解析struts.xml文件一些有用的值,以及反射時所調(diào)用的Action本身對象,其實就是一個簡單的JavaBean,存儲臨信息。
getActionClass(actionName)方法就是將URL上的actionName取出與struts.xml中的<action>結點匹配,可以得到Action的類全名,Action所調(diào)用的具體哪個方法名,Action的所有result結點(因為方法還沒有執(zhí)行,還不知道是具體哪一個result結點,所以先存起來,后面來解析)分別存在ActionClass對象中相應的屬性中去。具體的解析代碼我就不打出來了,否則文章太長。
再來看看setBeforeActionValue(clas, params)這一句。其實就是將得到的ActionClass對象與提交的參數(shù)全部傳進去,給Action的那些set屬性賦值:
/**
* 調(diào)用Action,并執(zhí)行Action的無參方法
*
* @param actionClass
* @param request.getParameterMap()
* @return
*/
public Object setActionValues(ActionClass actionClass,
Map<String, String[]> params) {
try {
// 得到Action的Class,并根據(jù)無參構造函數(shù)生成一個Action對象
Class clas = Class.forName(actionClass.getClassName());
Object obj = clas.newInstance();
if (params != null && params.size() > 0) {
Iterator<String> it = params.keySet().iterator();
while (it.hasNext()) {
String key = it.next();
String[] value = params.get(key);
String upperFirstLetter = key.substring(0, 1).toUpperCase();
// 獲得和屬性對應的setXXX()方法的名字
String setMethodName = "set" + upperFirstLetter
+ key.substring(1);
Method method = null;
// 看看該頁面提交的參數(shù)名中,是否在Action有set方法
try {
method = clas.getMethod(setMethodName,
new Class[] { String.class });
} catch (NoSuchMethodException e) {
System.out.println("警告 " + actionClass.getClassName()
+ "." + setMethodName + "("
+ String.class.getName() + ") 不存在");
}
if (method != null) {
// 如果有set方法,就調(diào)用set方法,進行賦值操作
String result = StringUtil.StringArrayToString(value);
method.invoke(obj, new String[] { result });
}
}
}
return obj;
......
}
這樣就順利的將頁面的值賦給了Action的相應屬性,接下來就是Action調(diào)用工作了。通過setActionValues方法,我們已經(jīng)可以得到Action對象本身了,可以存在ActionClass對象clas中去,我們直接調(diào)用setResultValue(clas, request)在Action執(zhí)行后,同時也把有get方法的屬性一并存于request域中:
/**
* 調(diào)用Action,并執(zhí)行Action的無參方法
*
* @param actionClass
* @param obj
* 要處理的對象
* @return
*/
public Object invokeAction(ActionClass actionClass) {
try {
Object obj = actionClass.getAction();
Class clas = obj.getClass();
Method method = clas.getMethod(actionClass.getMethod(), null);
String result = (String) method.invoke(obj, null);
this.setInvokeResult(result, actionClass);
actionClass.setAction(obj);
return obj;
......
}
}
很簡單的代碼——調(diào)用Action那個無參執(zhí)行方法,得到返回String類型的返回結果,然后我們可以再次利用ActionClass將最終的返回結果也給解析出來,看this.setInvokeResult(result, actionClass)方法
/**
* 匹配<result name="success">/index.jsp</result> Xml中的result
* name屬性,如果匹配成功,設置返回結果"/index.jsp"
*
* @param result
* @param actionClass
*/
private void setInvokeResult(String result, ActionClass actionClass) {
List<Element> elements = actionClass.getElements();
for (Element elem : elements) {
Attribute name = XmlUtil.getAttributeByName(elem, "name");
if (StringUtil.equals(result, name.getText())) {
actionClass.setResult(elem.getText());
return;
}
}
throw new RuntimeException("請確定在xml配置文件中是否有名叫 [" + result
+ "] 的返回類型結點 ");
}
一切大功告成,將剛剛得到的返回結果用servletContext.getRequestDispatcher(clas.getResult()).forward(
別忘記了在web.xml中配置這個Filter:
<filter> <filter-name>struts</filter-name> <filter-class>com.framework.core.StrutsFilter</filter-class> </filter> <filter-mapping> <filter-name>struts</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
最后寫個測試Action吧,就按照本文最開始的那個struts.xml配置編寫HelloAction.java
public class HelloAction {
private String message;
public String hello() {
message = "superleo " + this.message;
return "success";
}
public String getMessage() {
return message;
}
public void setMessage(String message) {
this.message = message;
}
}
如果下載了源代碼,可分別輸入下列幾個鏈接看看測試效果:(test文件夾下是單元測試) |
|
|