自己动手写一个Struts2 - superleo - JavaEye技术网站

来源:百度文库 编辑:神马文学网 时间:2024/04/28 11:50:30
自己动手写一个Struts2
关键字: struts2, diy, 自己动手
使用Struts2或webwork2有一段时间了,想把Struts2框架的思路简单的与大家分享一下,之前我是看过Struts2源代码的,所以本文算是它的一个功能非常有限的压缩版本。我也不打算重复发明轮子,只想让Struts2或Webwork2的新手更多的了解框架本身,而不仅仅是应用。废话少说,开始吧。
本文采用基本Xml来配置Action,如果有时间会继续写Annotation的实现。Xml文件结构与Struts2的配置文件struts.xml几乎一样,这样大家都比较熟悉,不过我简写了某些地方:
Struts.xml
Xml代码




class="com.leo.action.HelloAction">
/index.jsp



/index.jsp
是不是很熟悉啊,不过为了简单我都给简化了,否则这是一个没有尽头的工作。核心部分仍然是Filter,Struts2所有工作都是通过一个Filter来完成的(struts1.*是通过一个Action实现的)。我们先来看代码:
Java代码
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和传过来的参数
String actionName = StringUtil.parseServletPath(request
.getServletPath());
// 如果后缀不为.action,那么直接放过,不进行拦截
if (StringUtil.isEmpty(actionName)) {
chain.doFilter(request, response);
} else {
// 解析得到ActionClass,里面包括Action的类全名,返回页面值,Action执行的方法
ActionClass clas = this.getActionClass(actionName);
// 得到页面的所有parameters参数(没考虑上传情况)
Map params = request.getParameterMap();
// 为要调用的Action的set**方法设值,并返回要调用的Action对象本身
setBeforeActionValue(clas, params);
// 调用的Action执行方法,并返回值设置在request域中
setResultValue(clas, request);
// 返回相应的JSP页面
servletContext.getRequestDispatcher(clas.getResult()).forward(
request, response);
}
}
}
public class StrutsFilter implements Filter {public void doFilter(ServletRequest req, ServletResponse res,FilterChain chain) throws IOException, ServletException {// TODO Auto-generated method stubHttpServletRequest request = (HttpServletRequest) req;HttpServletResponse response = (HttpServletResponse) res;ServletContext servletContext = filterConfig.getServletContext();// 解析Request的URL和传过来的参数String actionName = StringUtil.parseServletPath(request.getServletPath());// 如果后缀不为.action,那么直接放过,不进行拦截if (StringUtil.isEmpty(actionName)) {chain.doFilter(request, response);} else {// 解析得到ActionClass,里面包括Action的类全名,返回页面值,Action执行的方法ActionClass clas = this.getActionClass(actionName);// 得到页面的所有parameters参数(没考虑上传情况)Map params = request.getParameterMap();// 为要调用的Action的set**方法设值,并返回要调用的Action对象本身setBeforeActionValue(clas, params);// 调用的Action执行方法,并返回值设置在request域中setResultValue(clas, request);// 返回相应的JSP页面servletContext.getRequestDispatcher(clas.getResult()).forward(request, response);}}}
没错一些因果都因doFilter方法而起。我的作法是:
解析URL路径,从而得到相应action在struts.xml配置文件中所配置的方法。如果不为合法的action后缀,直接chain.doFilter(request, response)放行。 将URL上的参数通过request.getParameterMap()取出,在调用action执行方法之前,将具有set**属性的字段赋值。这里利用了反射。 继续利用反射,执行Action的方法。结束前将具有get**属性的字段存于request域中,供页面使用。大家记得在每次执行Action的方法时都会返回一个String字符串,比如SUCCESS,INPUT,ERROR等,我们可以将这个值与struts.xml的/index.jsp 结点进行匹配,取出返回页面/index.jsp 一切准备就绪后,调用servletContext.getRequestDispatcher().forward()方法到相应的页面上去。
这就是大概的流程。因为是一个入门的框架所以很不完善,拦截器,result type的各种类型都没有去实现,因为我压根没想过要重复发明轮子。好,我们开始一步一步的看。
我们先来看这一句:ActionClass clas = this.getActionClass(actionName);
其中ActionClass的结构如下:
Java代码
public class ActionClass {
/**
* 类名
*/
private String className;
/**
* 要调用的方法名
*/
private String method;
/**
* 返回结果页面
*/
private String result;
/**
* 临时存储Action下的所有result结点
*/
private List elements = new ArrayList();
/**
* 要调用的Action本身
*/
private Object action;
//省略所有的set,get方法
}
public class ActionClass {/*** 类名*/private String className;/*** 要调用的方法名*/private String method;/*** 返回结果页面*/private String result;/*** 临时存储Action下的所有result结点*/private List elements = new ArrayList();/*** 要调用的Action本身*/private Object action;//省略所有的set,get方法}
ActionClass主要是用来存放解析struts.xml文件一些有用的值,以及反射时所调用的Action本身对象,其实就是一个简单的JavaBean,存储临信息。
getActionClass(actionName)方法就是将URL上的actionName取出与struts.xml中的结点匹配,可以得到Action的类全名,Action所调用的具体哪个方法名,Action的所有result结点(因为方法还没有执行,还不知道是具体哪一个result结点,所以先存起来,后面来解析)分别存在ActionClass对象中相应的属性中去。具体的解析代码我就不打出来了,否则文章太长。
再来看看setBeforeActionValue(clas, params)这一句。其实就是将得到的ActionClass对象与提交的参数全部传进去,给Action的那些set属性赋值:
Java代码
/**
* 调用Action,并执行Action的无参方法
*
* @param actionClass
* @param request.getParameterMap()
* @return
*/
public Object setActionValues(ActionClass actionClass,
Map params) {
try {
// 得到Action的Class,并根据无参构造函数生成一个Action对象
Class clas = Class.forName(actionClass.getClassName());
Object obj = clas.newInstance();
if (params != null && params.size() > 0) {
Iterator 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;
// 看看该页面提交的参数名中,是否在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方法,就调用set方法,进行赋值操作
String result = StringUtil.StringArrayToString(value);
method.invoke(obj, new String[] { result });
}
}
}
return obj;
......
}
/*** 调用Action,并执行Action的无参方法** @param actionClass* @param request.getParameterMap()* @return*/public Object setActionValues(ActionClass actionClass,Map params) {try {// 得到Action的Class,并根据无参构造函数生成一个Action对象Class clas = Class.forName(actionClass.getClassName());Object obj = clas.newInstance();if (params != null && params.size() > 0) {Iterator 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;// 看看该页面提交的参数名中,是否在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方法,就调用set方法,进行赋值操作String result = StringUtil.StringArrayToString(value);method.invoke(obj, new String[] { result });}}}return obj;......}
这样就顺利的将页面的值赋给了Action的相应属性,接下来就是Action调用工作了。通过setActionValues方法,我们已经可以得到Action对象本身了,可以存在ActionClass对象clas中去,我们直接调用setResultValue(clas, request)在Action执行后,同时也把有get方法的属性一并存于request域中:
Java代码
/**
* 调用Action,并执行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;
......
}
}
/*** 调用Action,并执行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;......}}
很简单的代码——调用Action那个无参执行方法,得到返回String类型的返回结果,然后我们可以再次利用ActionClass将最终的返回结果也给解析出来,看this.setInvokeResult(result, actionClass)方法
Java代码
/**
* 匹配/index.jsp Xml中的result
* name属性,如果匹配成功,设置返回结果"/index.jsp"
*
* @param result
* @param actionClass
*/
private void setInvokeResult(String result, ActionClass actionClass) {
List 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
+ "] 的返回类型结点 ");
}
/*** 匹配/index.jsp Xml中的result* name属性,如果匹配成功,设置返回结果"/index.jsp"** @param result* @param actionClass*/private void setInvokeResult(String result, ActionClass actionClass) {List 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(
request, response)转发出去,编码部分完毕。
别忘记了在web.xml中配置这个Filter:
Xml代码

struts
com.framework.core.StrutsFilter


struts
/*

strutscom.framework.core.StrutsFilterstruts/*
最后写个测试Action吧,就按照本文最开始的那个struts.xml配置编写HelloAction.java
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;
}
}
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文件夹下是单元测试)
http://localhost/framework/http://localhost/framework/hello.actionhttp://localhost/framework/hello.action?message=superleo