分享
 
 
 

设计模式:设计自己的MVC框架

王朝vc·作者佚名  2008-06-01
窄屏简体版  字體: |||超大  

源代码放在sharesources@126.com的邮箱的收件箱里,用户名:sharesource,密码:javafans

希望保留给有用的人,谢谢。

取这样一个标题太大,吸引眼球嘛@_@。

事实是最近读《J2EE设计模式》讲述表达层模式的那几章,书中有一个前端控制器+command模式的workflow例子,就琢磨着可以很简单地扩展成一个MVC框架。花了一个下午改写了下,对书中所述的理解更为深入。我想这也许对于学习和理解设计模式,以及初次接触struts等MVC框架的人可能有点帮助。因为整个模型类似于struts,我把它取名叫strutslet^_^。学习性质,切勿认真。

(一)完整的类图如下:

点击查看大图

1。前端控制器(FrontController):前端控制器提供了一个统一的位置来封装公共请求处理,它的任务相当简单,执行公共的任务,然后把请求转交给相应的控制器。在strutslet中,前端控制器主要作用也在于此,它初始化并解析配置文件,接受每个请求,并简单地把请求委托给调度器(Dispatcher),由调度器执行相应的动作(Action)。调度器把action返回的url返回给FrontController,FrontController负责转发。

2。Action接口:command模式很好的例子,它是一个命令接口,每一个实现了此接口的action都封装了某一个请求:新增一条数据记录并更新model,或者把某个文件写入磁盘。命令解耦了发送者和接受者之间联系。 发送者调用一个操作,接受者接受请求执行相应的动作,因为使用Command模式解耦,发送者无需知道接受者任何接口。

3。Dispatcher:调度器,负责流程的转发,负责调用action去执行业务逻辑。由调度器选择页面和action,它去除了应用行为和前端控制器间的耦合。调度器服务于前端控制器,它把model的更新委托给action,又提供页面选择给FrontController

4。ActionForward:封装了转向操作所需要信息的一个模型,包括name和转向url

5。ActionModel:解析配置文件后,将每一个Action封装成一个ActionModel对象,所有ActionModel构成一个map,并存储在ServletContext中,供整个框架使用。

(二)源代码简单分析

1。Action接口,只有一个execute方法,任何一个action都只要实现此接口,并实现相应的业务逻辑,最后返回一个ActionForward,提供给Dispacher调用。

public interface Action {

public ActionForward execute(HttpServletRequest request,ServletContext context);

}

比如,我们要实现一个登陆系统(demo的例子),LoginAction验证用户名和密码,假如正确,返回sUCcess页面,假如登陆失败,返回fail页面:

public class LoginAction implements Action {

PRivate String name="";

public ActionForward execute(HttpServletRequest request,

ServletContext context) {

String userName=request.getParameter("userName");

String passWord=request.getParameter("password");

if(userName.equals("dennis")&&password.equals("123")){

request.setAttribute("name", name);

return ActionForward.SUCCESS; //登陆成功,返回success

}else

return ActionForward.FAIL; //否则,返回fail

}

QQread.com 推出各大专业服务器评测 linux服务器的安全性能 SUN服务器 HP服务器 DELL服务器 IBM服务器 联想服务器 浪潮服务器 曙光服务器 同方服务器 华硕服务器 宝德服务器

2.还是先来看下两个模型:ActionForward和ActionModel,没什么东西,属性以及相应的getter,setter方法:

/**

* 类说明:转向模型

* @author dennis

*

* */

public class ActionForward {

private String name; //forward的name

private String viewUrl; //forward的url

public static final ActionForward SUCCESS=new ActionForward("success");

public static final ActionForward FAIL=new ActionForward("fail");

public ActionForward(String name){

this.name=name;

}

public ActionForward(String name, String viewUrl) {

super();

this.name = name;

this.viewUrl = viewUrl;

}

//...name和viewUrl的getter和setter方法

}

我们看到ActionForward预先封装了SUCCESS和FAIL对象。

public class ActionModel {

private String path; // action的path

private String className; // action的class

private Map<String, ActionForward> forwards; // action的forward

public ActionModel(){}

public ActionModel(String path, String className,

Map<String, ActionForward> forwards) {

super();

this.path = path;

this.className = className;

this.forwards = forwards;

}

//...相应的getter和setter方法

}

3。知道了两个模型是什么样,也应该可以猜到我们的配置文件大概是什么样的了,与struts的配置文件格式类似:

<?xml version="1.0" encoding="UTF-8"?>

<actions>

<action path="/login"

class="com.strutslet.demo.LoginAction">

<forward name="success" url="hello.jsp"/>

<forward name="fail" url="fail.jsp"/>

</action>

</actions>

path是在应用中将被调用的路径,class指定了调用的哪个action,forward元素指定了转向,比如我们这里假如是success就转向hello.jsp,失败的话转向fail.jsp,这里配置了demo用到的LoginAction。

4。Dispacher接口,主要是getNextPage方法,此方法负责获得下一个页面将导向哪里,提供给前端控制器转发。

public interface Dispatcher {

public void setServletContext(ServletContext context);

public String getNextPage(HttpServletRequest request,ServletContext context);

}

5。5。原先书中实现了一个WorkFlow的Dispatcher,按照顺序调用action,实现工作流调用。而我们所需要的是根据请求的path调用相应的action,执行action的execute方法返回一个ActionForward,然后得到ActionForward的viewUrl,将此viewUrl提供给前端控制器转发,看看它的getNextPage方法:

public String getNextPage(HttpServletRequest request, ServletContext context) {

setServletContext(context);

Map<String, ActionModel> actions = (Map<String, ActionModel>) context

.getAttribute(Constant.ACTIONS_ATTR); //从ServletContext得到所有action信息

String reqPath = (String) request.getAttribute(Constant.REQUEST_ATTR);//发起请求的path

ActionModel actionModel = actions.get(reqPath); //根据path得到相应的action

String forward_name = "";

ActionForward actionForward;

try {

Class c = Class.forName(actionModel.getClassName()); //每个请求对应一个action实例

Action action = (Action) c.newInstance();

actionForward = action.execute(request, context); //执行action的execute方法

forward_name = actionForward.getName();

} catch (Exception e) {

log.error("can not find action "+actionModel.getClassName());

e.printStackTrace();

}

actionForward = actionModel.getForwards().get(forward_name);

if (actionForward == null) {

log.error("can not find page for forward "+forward_name);

return null;

} else

return actionForward.getViewUrl(); //返回ActionForward的viewUrl

}

QQread.com 推出各大专业服务器评测 Linux服务器的安全性能 SUN服务器 HP服务器 DELL服务器 IBM服务器 联想服务器 浪潮服务器 曙光服务器 同方服务器 华硕服务器 宝德服务器

6。前端控制器(FrontController),它的任务我们已经很清楚,初始化配置文件;存储所有action到ServletContext供整个框架使用;得到发起请求的path,提供给Dispachter查找相应的action;调用Dispatcher,执行getNextPage方法得到下一个页面的url并转发:

public void init() throws ServletException {

//初始化配置文件

ServletContext context=getServletContext();

String config_file =getServletConfig().getInitParameter("config");

String dispatcher_name=getServletConfig().getInitParameter("dispatcher");

if (config_file == null config_file.equals(""))

config_file = "/WEB-INF/strutslet-config.xml"; //默认是/WEB-INF/下面的strutslet-config

if(dispatcher_name==nulldispatcher_name.equals(""))

dispatcher_name=Constant.DEFAULT_DISPATCHER;

try {

Map<String, ActionModel> resources = ConfigUtil.newInstance() //工具类解析配置文件

.parse(config_file, context);

context.setAttribute(Constant.ACTIONS_ATTR, resources); //存储在ServletContext中

log.info("初始化strutslet配置文件成功");

} catch (Exception e) {

log.error("初始化strutslet配置文件失败");

e.printStackTrace();

}

//实例化Dispacher

try{

Class c = Class.forName(dispatcher_name);

Dispatcher dispatcher = (Dispatcher) c.newInstance();

context.setAttribute(Constant.DISPATCHER_ATTR, dispatcher); //放在ServletContext

log.info("初始化Dispatcher成功");

}catch(Exception e) {

log.error("初始化Dispatcher失败");

e.printStackTrace();

}

.....

doGet()和doPost方法我们都让它调用process方法:

protected void process(HttpServletRequest request,

HttpServletResponse response) throws ServletException, IOException {

ServletContext context = getServletContext();

//获取action的path

String reqURI = request.getRequestURI();

int i=reqURI.lastIndexOf(".");

String contextPath=request.getContextPath();

String path=reqURI.substring(contextPath.length(),i);

request.setAttribute(Constant.REQUEST_ATTR, path);

Dispatcher dispatcher = (Dispatcher) context.getAttribute(Constant.DISPATCHER_ATTR);

// make sure we don't cache dynamic data

response.setHeader("Cache-Control", "no-cache");

response.setHeader("Pragma", "no-cache");

// use the dispatcher to find the next page

String nextPage = dispatcher.getNextPage(request, context);//调用Dispatcher的getNextPage

// forward control to the view

RequestDispatcher forwarder = request.getRequestDispatcher("/"

+ nextPage);

forwarder.forward(request, response); //转发页面

}

7。最后,web.xml的配置就非常简单了,配置前端控制器,提供启动参数(配置文件所在位置,为空就查找/WEB-INF/下面的strutslet-config.xml文件),我们把所有以action结尾的请求都交给FrontController处理:

<servlet>

<servlet-name>StrutsletController</servlet-name>

<servlet-class>com.strutslet.core.FrontController</servlet-class>

<!--

<init-param>

<param-name>config</param-name>

<param-value>/WEB-INFstrutslet-config.xml</param-value>

</init-param>

-->

<load-on-startup>0</load-on-startup>

</servlet>

<servlet-mapping>

<servlet-name>StrutsletController</servlet-name>

<url-pattern>*.action</url-pattern>

</servlet-mapping>

最后,让我们看看整个框架图:

点击查看大图

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有