設計自己的MVC框架(1)

轉自 http://www.blogjava.net/killme2008/archive/2007/02/06/98227.html

事實是最近讀《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 
 
 } 

 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章