设计自己的MVC框架

简介:

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

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

(一)完整的类图如下:

      10fd68bb171.jpg

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调用。

 

  package  com.strutslet.core;

  import  javax.servlet.ServletContext;
  import  javax.servlet.http.HttpServletRequest;

  import  com.strutslet.model.ActionForward;

  /** 
 * command接口
 * 
 @author  dennis
 *
 
 */ 

  public    interface  Action   {
  public  ActionForward execute(HttpServletRequest request,ServletContext context); 

 

 

比如,我们要实现一个登陆系统,LoginAction验证用户名和密码,如果正确,返回success页面,如果登陆失败,返回fail页面:

  package  com.strutslet.demo;

  import  javax.servlet.ServletContext;
  import  javax.servlet.http.HttpServletRequest;

  import  com.strutslet.core.Action;
  import  com.strutslet.model.ActionForward;

  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 
 
 } 

 

 


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

  package  com.strutslet.model;

  /** 
 * 类说明:转向模型
 * 
 @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对象。

  //  ActionModel.java 
 

  package  com.strutslet.model;

  import  java.util.Map;

  /** 
 * 类说明:
 * 
 @author  dennis
 *
 
 */ 

 
  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方法,此方法负责获得下一个页面将导向哪里,提供给前端控制器转发。

  package  com.strutslet.core;


  import  javax.servlet.ServletContext;
  import  javax.servlet.http.HttpServletRequest;

  /** 
 * service to worker模式,提供给FrontController使用
 * 负责流程转发
 * 
 @author  dennis
 *
 
 */ 

  public    interface  Dispatcher   {
  public   void  setServletContext(ServletContext context);
  public  String getNextPage(HttpServletRequest request,ServletContext context);

 


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 
 
 } 

 


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 == null || dispatcher_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 - INF / strutslet - 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 >
文章转自庄周梦蝶  ,原文发布5.16
目录
相关文章
|
前端开发 Java 程序员
从零基础手写Spring MVC框架,准备好进阶程序员了吗?
我们程序员大部分人都是野路子,不懂什么叫代码规范。写了一个月的代码,最后还得其他老司机花3天时间重构,相信大部分老司机都很头疼看新手的代码。
123 1
|
设计模式 前端开发 JavaScript
浅谈MVC、MVP、MVVM框架模式
浅谈MVC、MVP、MVVM框架模式
140 0
|
前端开发 数据库 Python
Python Web 开发: 解释 Django 框架的 MVC 架构是什么?
Python Web 开发: 解释 Django 框架的 MVC 架构是什么?
491 0
|
Java 数据库连接 Maven
SSM框架整合:掌握Spring+Spring MVC+MyBatis的完美结合!
SSM框架整合:掌握Spring+Spring MVC+MyBatis的完美结合!
|
9月前
|
前端开发 Java 数据库连接
Spring MVC 扩展和SSM框架整合
通过以上步骤,我们可以将Spring MVC扩展并整合到SSM框架中。这个过程包括配置Spring MVC和Spring的核心配置文件,创建控制器、服务层和MyBatis的Mapper接口及映射文件。在实际开发中,可以根据具体业务需求进行进一步的扩展和优化,以构建更加灵活和高效的企业级应用程序。
229 5
|
12月前
|
监控 前端开发 API
一款基于 .NET MVC 框架开发、功能全面的MES系统
一款基于 .NET MVC 框架开发、功能全面的MES系统
354 5
|
前端开发 Java
【案例+源码】详解MVC框架模式及其应用
【案例+源码】详解MVC框架模式及其应用
1138 0
|
设计模式 存储 前端开发
MVC 框架的主要问题是什么?
【8月更文挑战第30天】
304 0
|
前端开发 安全 Java
使用Java Web框架:Spring MVC的全面指南
【4月更文挑战第3天】Spring MVC是Spring框架的一部分,用于构建高效、模块化的Web应用。它基于MVC模式,支持多种视图技术。核心概念包括DispatcherServlet(前端控制器)、HandlerMapping(请求映射)、Controller(处理请求)、ViewResolver(视图解析)和ModelAndView(模型和视图容器)。开发流程涉及配置DispatcherServlet、定义Controller、创建View、处理数据、绑定模型和异常处理。
563 1
使用Java Web框架:Spring MVC的全面指南
|
前端开发 Java Spring
Java Web ——MVC基础框架讲解及代码演示(下)
Java Web ——MVC基础框架讲解及代码演示
168 1