手动的写一个structs

为了更好的学习框架的运行机制,这里开始学习框架之前,介绍一个简单的自定义的框架。

需求:

  登录:id:aaa,pwd:888登录成功之后,跳转到,index.jsp页面并显示,欢迎你,aaa

  注册,页面,输入用户名密码,点击注册。注册成功之后,将会跳转到登录界面。

  重在了解前后的这个逻辑,所以把后天是写死的。

entity层

  就一个User

package cn.itcast.entity;

public class User {

    private String name;
    private String pwd;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getPwd() {
        return pwd;
    }
    public void setPwd(String pwd) {
        this.pwd = pwd;
    }

}

接着是到层

  就一个UserDao,模拟了注册,登录,没有都是写死的

 1 package cn.itcast.dao;
 2
 3 import cn.itcast.entity.User;
 4
 5
 6 public class UserDao {
 7
 8     // 模拟登陆
 9     public User login(User user){
10         if ("tom".equals(user.getName()) && "888".equals(user.getPwd()) ){
11             // 登陆成功
12             return user;
13         }
14         // 登陆失败
15         return null;
16     }
17
18     // 模拟注册
19     public void register(User user) {
20         System.out.println("注册成功:用户," + user.getName());
21     }
22 }

service层还和以前一样,没有什么区别,直接使用就可以了这里也就一个UserService

 1 package cn.itcast.service;
 2
 3 import cn.itcast.dao.UserDao;
 4 import cn.itcast.entity.User;
 5
 6
 7 public class UserService {
 8
 9     private UserDao ud = new UserDao();
10
11     // 模拟登陆
12     public User login(User user){
13         return ud.login(user);
14     }
15
16     // 模拟注册
17     public void register(User user) {
18         ud.register(user);
19     }
20 }

framewoek层,

  这是手动缩写的这个,mystructs的核心部分。

  以往在ervlet层中写servlet,反观一下,servlet层在mvc中负责control的角色:获取参数,调用service,跳转页面。无非就这三大块。

  不同的页面请求提交到不同的servt,但最终从宏观上来说无非是上面的三块内容。于是我们想做到的是,值写一个servlet这里我们命名为ActionServlet,这个类的功能,就是统筹全局请求的分配,针对比如像http:localhost:8080/mystucets/login.action的请求,ActionServlet解析出login,把这个请求,交给LoginAction类处理,类似于register.action的请求交给对应的RegisterAction处理。  

  要想完成这种准确无误的转发关系,或者说是一种映射。那么ActionServlet要一个根据,这个根据就是一个ActionSerlet可以查找的表。我们先来看一个,上面设想的架构的时序流

1.就收请求,http:localhost:8080/mystucets/login.action,解析出login

2。转发给LoginAction处理,LoginAction调用login()方法,这个方法执行完成之后返回一个returnFlag标志,loginSuccess表示登陆成功,loginFial表示登录失败,

3.ActionServlet根据返回的returnFlag判断跳转的页面,当然要先查询配置(映射表)文件,获取对应的跳转page,以及跳转的方式(转发或者重定向)

我们需要一个配置文件,这个配置文件是一个xml文件,这个文件我们根据,login或者register找到所对应的Action类,进一步找到跳转的可能,设计这个mystructs.xml的配置文件如下:

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <mystruts>
 3     <package>
 4         <!-- 配置请求路径,与处理action类的关系 -->
 5         <!--
 6             1. 请求路径与处理Action的关系
 7                  /login = LoginAction                          login
 8                         success = /index.jsp                     登陆成功(重定向)
 9                         loginFaild  = /login.jsp                 登陆失败
10
11          -->
12         <action name="login" class="cn.itcast.framework.action.LoginAction" method="login">
13             <result name="loginSuccess" type="redirect">/index.jsp</result>
14             <result name="loginFaild">/login.jsp</result>
15         </action>
16
17         <action name="register" class="cn.itcast.framework.action.RegisterAction" method="register">
18             <result name="registerSuccess">/login</result>
19         </action>
20
21     </package>
22
23 </mystruts>

  对于一个项目而言,要处理的请求有很多,也就说有很多的XxxAction,这是需要我们,那么对应的mystructs.xml文件就会很大,我们不会每次都来查找解析这个xml文件,而且对于需要同时管理很多信息时,常见的做法是把这若干信息封装到一些bean中,讲分页的时候已经使用过这种策略了。每个<action>封装到对应的ActionMapping类中,每个<result>封装到Result类中,因为每个<action>中包含若干的<result>所以每个ActionMapping总包含若干的<Result>。ActionMapping和Result的设计如下:

ActionMapping:

 1 package cn.itcast.framework.bean;
 2
 3 import java.util.Map;
 4
 5 /**
 6  * 封装action节点
 7  *      <action name="login" class="cn.itcast.framework.action.LoginAction" method="login">
 8             <result name="success" type="redirect">/index.jsp</result>
 9             <result name="loginFaild">/login.jsp</result>
10         </action>
11
12  * @author Jie.Yuan
13  *
14  */
15 public class ActionMapping {
16
17     // 请求路径名称
18     private String name;
19     // 处理aciton类的全名
20     private String className;
21     // 处理方法
22     private String method;
23     // 结果视图集合
24     private Map<String,Result> results;
25
26     public String getName() {
27         return name;
28     }
29     public void setName(String name) {
30         this.name = name;
31     }
32     public String getClassName() {
33         return className;
34     }
35     public void setClassName(String className) {
36         this.className = className;
37     }
38     public String getMethod() {
39         return method;
40     }
41     public void setMethod(String method) {
42         this.method = method;
43     }
44     public Map<String, Result> getResults() {
45         return results;
46     }
47     public void setResults(Map<String, Result> results) {
48         this.results = results;
49     }
50
51
52
53 }

Result:

 1 package cn.itcast.framework.bean;
 2
 3 /**
 4  * 封装结果视图
 5  * <result name="success" type="redirect">/index.jsp</result>
 6  * @author Jie.Yuan
 7  *
 8  */
 9 public class Result {
10
11     // 跳转的结果标记
12     private String name;
13     // 跳转类型,默认为转发; "redirect"为重定向
14     private String type;
15     // 跳转的页面
16     private String page;
17     public String getName() {
18         return name;
19     }
20     public void setName(String name) {
21         this.name = name;
22     }
23     public String getType() {
24         return type;
25     }
26     public void setType(String type) {
27         this.type = type;
28     }
29     public String getPage() {
30         return page;
31     }
32     public void setPage(String page) {
33         this.page = page;
34     }
35
36 }

为了避免每次查询mystructs.xml问价(这样使很消耗时间的),我们的做法是在首次访问ActionServlet的时候,也就是在ActionServlet的init()函数中把mystructs.xml中的内容全部读取出来(在后面的ActionServlet的代码可以看到),读取出来之后我们使用一个ActionMappingManager进行管理,下面是ActionMappingManager类:

  1 package cn.itcast.framework.bean;
  2
  3 import java.io.InputStream;
  4 import java.util.HashMap;
  5 import java.util.Iterator;
  6 import java.util.List;
  7 import java.util.Map;
  8
  9 import org.dom4j.Document;
 10 import org.dom4j.Element;
 11 import org.dom4j.io.SAXReader;
 12
 13 /**
 14  * 加载配置文件, 封装所有的真个mystruts.xml
 15  * @author Jie.Yuan
 16  *
 17  */
 18 public class ActionMappingManager {
 19
 20     // 保存action的集合
 21     private Map<String,ActionMapping> allActions ;
 22
 23     public ActionMappingManager(){
 24         allActions = new HashMap<String,ActionMapping>();
 25         // 初始化
 26         this.init();
 27     }
 28
 29     /**
 30      * 根据请求路径名称,返回Action的映射对象
 31      * @param actionName   当前请求路径
 32      * @return             返回配置文件中代表action节点的AcitonMapping对象
 33      */
 34     public ActionMapping getActionMapping(String actionName) {
 35         if (actionName == null) {
 36             throw new RuntimeException("传入参数有误,请查看struts.xml配置的路径。");
 37         }
 38
 39         ActionMapping actionMapping = allActions.get(actionName);
 40         if (actionMapping == null) {
 41             throw new RuntimeException("路径在struts.xml中找不到,请检查");
 42         }
 43         return actionMapping;
 44     }
 45
 46     // 初始化allActions集合
 47     private void init() {
 48         /********DOM4J读取配置文件***********/
 49         try {
 50             // 1. 得到解析器
 51             SAXReader reader = new SAXReader();
 52             // 得到src/mystruts.xml  文件流
 53             InputStream inStream = this.getClass().getResourceAsStream("/mystruts.xml");
 54             // 2. 加载文件
 55             Document doc = reader.read(inStream);
 56
 57             // 3. 获取根
 58             Element root = doc.getRootElement();
 59
 60             // 4. 得到package节点
 61             Element ele_package = root.element("package");
 62
 63             // 5. 得到package节点下,  所有的action子节点
 64             List<Element> listAction = ele_package.elements("action");
 65
 66             // 6.遍历 ,封装
 67             for (Element ele_action : listAction) {
 68                 // 6.1 封装一个ActionMapping对象
 69                 ActionMapping actionMapping = new ActionMapping();
 70                 actionMapping.setName(ele_action.attributeValue("name"));
 71                 actionMapping.setClassName(ele_action.attributeValue("class"));
 72                 actionMapping.setMethod(ele_action.attributeValue("method"));
 73
 74                 // 6.2 封装当前aciton节点下所有的结果视图
 75                 Map<String,Result> results = new HashMap<String, Result>();
 76
 77                 // 得到当前action节点下所有的result子节点
 78                  Iterator<Element> it = ele_action.elementIterator("result");
 79                  while (it.hasNext()) {
 80                      // 当前迭代的每一个元素都是 <result...>
 81                      Element ele_result = it.next();
 82
 83                      // 封装对象
 84                      Result res = new Result();
 85                      res.setName(ele_result.attributeValue("name"));
 86                      res.setType(ele_result.attributeValue("type"));
 87                      res.setPage(ele_result.getTextTrim());
 88
 89                      // 添加到集合
 90                      results.put(res.getName(), res);
 91                  }
 92
 93                 // 设置到actionMapping中
 94                 actionMapping.setResults(results);
 95
 96                 // 6.x actionMapping添加到map集合
 97                 allActions.put(actionMapping.getName(), actionMapping);
 98             }
 99
100
101         } catch (Exception e) {
102             throw new RuntimeException("启动时候初始化错误",e);
103         }
104     }
105 }

  我们说了,ActionServlet位居中枢相当于行军主将,居中调遣需要冲锋陷阵的士卒(XxxAction),这里就是LoginAction和RegisterAction,下面是他们的定义:

  LoginAction:

 1 package cn.itcast.framework.action;
 2
 3 import java.io.IOException;
 4
 5 import javax.servlet.ServletException;
 6 import javax.servlet.http.HttpServletRequest;
 7 import javax.servlet.http.HttpServletResponse;
 8
 9 import cn.itcast.entity.User;
10 import cn.itcast.service.UserService;
11
12 /**
13  * Action表示动作类 1. 一个servlet对应一个action 2. action中负责处理具体的请求
14  *
15  * @author Jie.Yuan
16  *
17  */
18 public class LoginAction {
19
20     public Object execute(HttpServletRequest request, HttpServletResponse response)
21     throws ServletException, IOException {
22         return null;
23     }
24
25     /**
26      * 处理登陆请求
27      */
28     public Object login(HttpServletRequest request, HttpServletResponse response)
29             throws ServletException, IOException {
30         Object uri = null;
31
32         // 1. 获取请求数据,封装
33         String name = request.getParameter("name");
34         String pwd = request.getParameter("pwd");
35         User user = new User();
36         user.setName(name);
37         user.setPwd(pwd);
38
39         // 2. 调用Service
40         UserService userService = new UserService();
41         User userInfo = userService.login(user);
42         // 3. 跳转
43         if (userInfo == null) {
44             // 登陆失败
45 //            request.getRequestDispatcher("/login.jsp").forward(request,
46 //                    response);
47 //            uri = request.getRequestDispatcher("/login.jsp");
48             uri = "loginFaild";   // loginFaild  = /login.jsp
49         } else {
50             // 登陆成功
51             request.getSession().setAttribute("userInfo", userInfo);
52             // 首页
53 //            response.sendRedirect(request.getContextPath() + "/index.jsp");
54 //            uri = "/index.jsp";
55             uri = "loginSuccess";  // loginSuccess = /index.jsp
56         }
57         return uri;
58     }
59 }

  Register类:

package cn.itcast.framework.action;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import cn.itcast.entity.User;
import cn.itcast.service.UserService;

public class RegisterAction {

    public Object register(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        Object uri;

        // 1. 获取请求数据,封装
        String name = request.getParameter("name");
        String pwd = request.getParameter("pwd");
        User user = new User();
        user.setName(name);
        user.setPwd(pwd);

        // 2. 调用Service
        UserService userService = new UserService();
        userService.register(user);
        // 3. 跳转
//        request.getRequestDispatcher("/login.jsp").forward(request, response);
        //uri = request.getRequestDispatcher("/login.jsp");
        return "registerSuccess"; //返回注册的标记;   registerSuccess = /login.jsp

    }
}

  中军主帅ActionServlet类:

 1 package cn.itcast.framework;
 2
 3 import java.io.IOException;
 4 import java.lang.reflect.Method;
 5
 6 import javax.servlet.ServletException;
 7 import javax.servlet.http.HttpServlet;
 8 import javax.servlet.http.HttpServletRequest;
 9 import javax.servlet.http.HttpServletResponse;
10
11 import cn.itcast.framework.bean.ActionMapping;
12 import cn.itcast.framework.bean.ActionMappingManager;
13 import cn.itcast.framework.bean.Result;
14
15 /**
16  * 核心控制器,此项目只有这一个servlet
17  * 1. 拦截所有的*.action为后缀的请求
18  * 2. 请求:http://localhost:8080/mystruts/login.action
19  *           http://localhost:8080/mystruts/register.action
20
21  * @author Jie.Yuan
22  *
23  */
24 public class ActionServlet extends HttpServlet{
25
26     private ActionMappingManager actionMappingManager;
27
28     // 只执行一次  (希望启动时候执行)
29     @Override
30     public void init() throws ServletException {
31         System.out.println("1111111111111111ActionServlet.init()");
32         actionMappingManager = new ActionMappingManager();
33     }
34
35     // http://localhost:8080/mystruts/login.action
36     @Override
37     protected void doGet(HttpServletRequest request, HttpServletResponse response)
38             throws ServletException, IOException {
39
40         try {
41             // 1. 获取请求uri, 得到请求路径名称   【login】
42             String uri = request.getRequestURI();
43             // 得到 login
44             String actionName=uri.substring(uri.lastIndexOf("/")+1, uri.indexOf(".action"));
45
46             // 2. 根据路径名称,读取配置文件,得到类的全名   【cn..action.LoginAction】
47             ActionMapping actionMapping = actionMappingManager.getActionMapping(actionName);
48             String className = actionMapping.getClassName();
49
50             // 当前请求的处理方法   【method="login"】
51             String method = actionMapping.getMethod();
52
53             // 3. 反射: 创建对象,调用方法; 获取方法返回的标记
54             Class<?> clazz = Class.forName(className);
55             Object obj = clazz.newInstance();  //LoginAction loginAction = new LoginAction();
56             Method m = clazz.getDeclaredMethod(method, HttpServletRequest.class,HttpServletResponse.class );
57             // 调用方法返回的标记
58             String returnFlag =  (String) m.invoke(obj, request, response);
59
60             // 4. 拿到标记,读取配置文件得到标记对应的页面 、 跳转类型
61             Result result = actionMapping.getResults().get(returnFlag);
62             // 类型
63             String type = result.getType();
64             // 页面
65             String page = result.getPage();
66
67             // 跳转
68             if ("redirect".equals(type)) {
69                 response.sendRedirect(request.getContextPath() + page);
70             } else {
71                 request.getRequestDispatcher(page).forward(request, response);
72             }
73         } catch (Exception e) {
74             e.printStackTrace();
75         }
76     }
77
78
79     @Override
80     protected void doPost(HttpServletRequest req, HttpServletResponse resp)
81             throws ServletException, IOException {
82         doGet(req, resp);
83     }
84 }

注意了,这里只有一个ActionServlet处理所有的页面请求,将处理所有的http:localhost:8080/mystructs/*.action请求,那么就要在web.xml中配置ActionServlet的<pattern-url>时使用模糊匹配具体如下:

 1 <!-- 核心控制器 -->
 2   <servlet>
 3     <servlet-name>ActionServlet</servlet-name>
 4     <servlet-class>cn.itcast.framework.ActionServlet</servlet-class>
 5     <!-- 启动时候执行servlet初始化方法 -->
 6     <load-on-startup>1</load-on-startup>
 7   </servlet>
 8   <servlet-mapping>
 9     <servlet-name>ActionServlet</servlet-name>
10     <url-pattern>*.action</url-pattern>
11   </servlet-mapping>

为了保证启动时执行ActionServlet的init()方法,还用了一个参数<load-on-startup>1</load-on-startup>,使用这个参数,init函数将会在启动服务器之后调用,不然的第一次访问ActionServlet,ActionServlet创建对象时才会执行init函数。

到这里,这个简单的模拟项目就结束了,就可以开始测试了。我们访问http://localhost:8080/mystructs/login.jsp,然后输入用户名aaa和密码888,点击登录于是请求login.action,然后跳转到index.jsp页面看到欢迎你,aaa

时间: 2024-08-25 06:06:42

手动的写一个structs的相关文章

原来热加载如此简单,手动写一个 Java 热加载吧

1. 什么是热加载 热加载是指可以在不重启服务的情况下让更改的代码生效,热加载可以显著的提升开发以及调试的效率,它是基于 Java 的类加载器实现的,但是由于热加载的不安全性,一般不会用于正式的生产环境. 2. 热加载与热部署的区别 首先,不管是热加载还是热部署,都可以在不重启服务的情况下编译/部署项目,都是基于 Java 的类加载器实现的. 那么两者到底有什么区别呢? 在部署方式上: 热部署是在服务器运行时重新部署项目. 热加载是在运行时重新加载 class. 在实现原理上: 热部署是直接重新

为PhoneGap写一个android插件

为PhoneGap写一个android插件,要怎么做? 其实这句话应该反过来说,为android写一个PhoneGap插件,要怎么做? 这里以最简单的Hello World!为例,做个说明: 1.第一步,要先建立一个支持PhoneGap(Cordova)的android工程 因为这个插件本质上是安卓插件,用于PhoneGap,因此,要二者支持才行,所以我们要建立一个支持PhoneGap(Cordova)的android工程,插件在这个工程里面编写. 扫盲:PhoneGap现在已经出售给了Apac

linux设备驱动第三篇:写一个简单的字符设备驱动

在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存. 下面就开始学习如何写一个简单的字符设备驱动.首先我们来分解一下字符设备驱动都有那些结构或者方法组成,也就是说实现一个可以使用的字符设备驱动我们必须做些什么工作. 1.主设备号和次设备号 对于字符设备的访问是通过文件系统中的设备名称进行的.他们通常位于/dev目录下.如下: [plain] vie

linux设备驱动第三篇:如何写一个简单的字符设备驱动?

在linux设备驱动第一篇:设备驱动程序简介中简单介绍了字符驱动,本篇简单介绍如何写一个简单的字符设备驱动.本篇借鉴LDD中的源码,实现一个与硬件设备无关的字符设备驱动,仅仅操作从内核中分配的一些内存. 下面就开始学习如何写一个简单的字符设备驱动.首先我们来分解一下字符设备驱动都有那些结构或者方法组成,也就是说实现一个可以使用的字符设备驱动我们必须做些什么工作. 1.主设备号和次设备号 对于字符设备的访问是通过文件系统中的设备名称进行的.他们通常位于/dev目录下.如下: [email prot

用node写一个皖水公寓自动刷房源脚本

因为住的地方离公司太远,每天上下班都要坐很久的班车,所以最近想搬到公司旁边的皖水公寓住.去问了一下公寓的客服,客服说房源现在没有了,只能等到别人退房,才能在网站上申请到. 如果纯靠手动F5刷新浏览器,来捡漏可入住房源,那简直是太难了.要知道很多人都在电脑前面按着F5,感觉和春运抢票一样. 所以就准备写一个脚本来监测房源,解放双手. 平时对node.js用的比较多,所以就用node.js来写,怎么方便咱怎么搞. 一.房源信息抓取 通过对公寓申请网站的数据分析,找到了房源列表的接口地址. 二.数据获

继承?静态代理?写一个自己的动态代理吧

[ 需求分析 ] 在我们实际开发中常常会遇到这样的问题:记录一个类的方法运行时间,以分析性能.一般我们的做法是先在类的开始记录一个开始时间,然后在类的结束记录一个结束时间,二者相减就可以获取我们想要的结果.但是很多时候这些类已经打了jar包,我们无法直接修改源码,这个时候我们应该怎么办呢? 下文使用Tank的移动需要统计时间.记录日志来模拟需求场景,假定Moveable.Tank类无法修改. interface:Moveable public interface Moveable { publi

一起写一个JSON解析器

[本篇博文会介绍JSON解析的原理与实现,并一步一步写出来一个简单但实用的JSON解析器,项目地址:SimpleJSON.希望通过这篇博文,能让我们以后与JSON打交道时更加得心应手.由于个人水平有限,叙述中难免存在不准确或是不清晰的地方,希望大家可以指正:)] 一.JSON解析器介绍 相信大家在平时的开发中没少与JSON打交道,那么我们平常使用的一些JSON解析库都为我们做了哪些工作呢?这里我们以知乎日报API返回的JSON数据来介绍一下两个主流JSON解析库的用法.我们对地址 http://

python 学习笔记 12 -- 写一个脚本获取城市天气信息

最近在玩树莓派,前面写过一篇在树莓派上使用1602液晶显示屏,那么能够显示后最重要的就是显示什么的问题了.最容易想到的就是显示时间啊,CPU利用率啊,IP地址之类的.那么我觉得呢,如果能够显示当前时间.温度也是甚好的,作为一个桌面小时钟还是很精致的. 1. 目前有哪些工具 目前比较好用的应该是 weather-util, 之前我获取天气信息一般都是通过它. 使用起来也很简单: (1) Debian/Ubuntu 用户使用 sudo apt-get install weather-util 安装

爬虫入门 手写一个Java爬虫

本文内容 涞源于  罗刚 老师的 书籍 << 自己动手写网络爬虫一书 >> ; 本文将介绍 1: 网络爬虫的是做什么的?  2: 手动写一个简单的网络爬虫; 1: 网络爬虫是做什么的?  他的主要工作就是 跟据指定的url地址 去发送请求,获得响应, 然后解析响应 , 一方面从响应中查找出想要查找的数据,另一方面从响应中解析出新的URL路径, 然后继续访问,继续解析;继续查找需要的数据和继续解析出新的URL路径  . 这就是网络爬虫主要干的工作.  下面是流程图: 通过上面的流程图