一 开文背景 -- 废话讲一段~
本文借助动力节点-王勇老师的视频教程中的引例来了解struts1的实现原理,虽然现在已经很少使用struts1了,但是了解了其原理之后,对了解其他mvc框架还是有较大的帮助的.
二 简介 -- 切入主题
struts1主要实现从请求到servlet的映射.
e.g. 现有的一个业务为实现用户的添加删除修改,按照原有的model2的原理可以实现上述功能.
2.1没有struts1之前的基于model2 mvc的实现.
直接在servlet中实现如下代码:
String username = request.getParameter("username");
UserManager userManager = new UserManager();
String forward = "";
if ("/servlet/delUser".equals(path)) {
userManager.del(username);
forward = "/del_success.jsp";
}else if ("/servlet/addUser".equals(path)) {
userManager.add(username);
forward = "/add_success.jsp";
}else if ("/servlet/modifyUser".equals(path)) {
userManager.modify(username);
forward = "/modify_success.jsp";
}else if ("/servlet/queryUser".equals(path)) {
List userList = userManager.query(username);
request.setAttribute("userList", userList);
forward = "/query_success.jsp";
}else {
throw new RuntimeException("请求失败");
}
request.getRequestDispatcher(forward).forward(request, response);
到现在,基本的功能是已经实现了,但是看着上面的代码就像吐啊~一坨的if-else,复杂的每一个if-else节点处理~毫无半点的扩展性可言.
解决办法:
将每一个小的业务处理单独来处理.抽象出一个借口Action,抽象方法:execute(HttpServletRequest,HttpServletResponse). struts1中返回一个ActionForword对象.
public interface Action {
public String execute(HttpServletRequest request, HttpServletResponse response)
throws Exception;
}
每一个小的功能(添加\删除\修改),都抽象成一个action,实现上面的接口.
Action action = null;
if ("/servlet/delUser".equals(path)) {
action = new DelUserAction();
}else if ("/servlet/addUser".equals(path)) {
action = new AddUserAction();
}else if ("/servlet/modifyUser".equals(path)) {
action = new ModifyUserAction();
}else if ("/servlet/queryUser".equals(path)) {
action = new QueryUserAction();
}else {
throw new RuntimeException("请求失败");
}
String forward = null;
try {
forward = action.execute(request, response);
} catch (Exception e) {
e.printStackTrace();
}
request.getRequestDispatcher(forward).forward(request, response);
虽然相对上面的"代码",这次有了一定程度的提高(仅仅是思想上),代码的扩展性,还是没有达到要求的.
如今,对一些较多可选择性的代码,将其从代码中抽离出来,转化为配置文件是提高扩展性的一种方式.
/
<action-config>
<action path="/servlet/delUser" type="com.bjpowernode.servlet.DelUserAction">
<forward name="success">/del_success.jsp</forward>
<forward name="error">/del_error.jsp</forward>
</action <action path="/servlet/addUser" type="com.bjpowernode.servlet.AddUserAction">
<forward name="success">/add_success.jsp</forward>
<forward name="error">/add_error.jsp</forward>
</action <action path="/servlet/modifyUser" type="com.bjpowernode.servlet.ModifyUserAction">
<forward name="success">/modify_success.jsp</forward>
<forward name="error">/modify_error.jsp</forward>
</action <action path="/servlet/queryUser" type="com.bjpowernode.servlet.QueryUserAction">
<forward name="success">/query_success.jsp</forward>
<forward name="error">/query_error.jsp</forward>
</action </action-config>
每一个<action></action>结点指定该action对应的请求路径,对应处理该请求的具体Action类型,以及处理之后的转向.在装载这些配置文件需要一个对象来装填,这就是ActionMapping类型.
/
ActionMapping {
private String path;//请求路径
private String type;//对应的处理该请求的action类型.
Map forwardMap; //处理完成之后的跳转信息.
/
同样,需要将跳转信息保存起来,实现对象为ForwordMap<key,value>
/
}
forwardMap {
key="success";
value="/del_success.jsp"
key="error"
value="/del_error.jsp"
}
/
在struts1中使用Map将这些action保存起来Map<request_url,ActionMapping>.在actionmapping中同样保存着request_url.
Map map = new HashMap();
map.put("/servlet/delUser", actionMapping1);
map.put("/servlet/addUser", actionMapping2);
map.put("/servlet/modifyUser", actionMapping3);
map.put("/servlet/queryUser", actionMapping4);
如果是删除ActionMapping存储如下:
actionMapping {
path= "/servlet/delUser";
type = "com.bjpowernode.servlet.DelUserAction";
forwardMap {
key="success",value="/del_success.jsp"
key="error", value="/del_error.jsp"
}
}
三 综合总结
下面就使用一个例子来总结一下整个struts1的处理流程
String path = "/servlet/delUser";
1.根据截取的URL请求,到Map(从配置文件中获取)中取得本次请求对应的Action
ActionMappint actionMappint = (ActionMappint)map.get(path);
2.取得本请求对应的Action类的完整路径
String type = actionMappint.getType(); //com.bjpowernode.servlet.DelUserAction
3.采用反射动态实例化Action
Action action = (Action)class.forName(type).newInstance();
4.动态待用Action中的execute方法
String forward = action.execute(request, response);
5.根据路径完成转向
request.getRequestDispatcher(forward).forward(request, response);
真正来驱动这一切行为的操作者为Servlet,也就是MVC中的控制器,实现了请求url的截取,然后从配置文件中按照截取的url,来实现分发.读取配置文件,创建actionmapping,forwardmap对象,由此创建action对象,实现业务的处理,然后转向.