Servlet请求分派器——DispatcherServlet

章节索引

  • 前提
  • 先从Volley说起
  • 借鉴简单工厂模式的思想
  • 开始操作
    • DispatcherServlet
    • web.xml中的配置
    • 客户端
  • 后记


前提

在索引文章中我提到过,我会写一篇关于Servlet的进阶知识,这篇就是了。学习这篇文章之前,请先把前面整套帖子中的程序跑通,这篇文章是作为进阶补充的。


先从Volley说起

一般情况下,我们在Volley请求中书写的Servlet URL如以下格式:

//域名形式
String url = "http://implementist.com/myfirstapp/LoginServlet";//IP形式
String url = "http://1.1.1.1/myfirstapp/LoginServlet";

项目写大了以后,Servlet越来越多,不仅控制起来很不方便,且以后想要维护时,如果我改了服务器端Servlet的名字,客户端还得一个一个改URL。一次我们需要一个可维护性较高的替代方法。


借鉴简单工厂模式的思想

学过设计模式的同学应该很熟悉工厂模式,没学过的也没关系。

其大体的思想是:如果我们具有多个同级相似的类(如加法类、减法类、乘法类、除法类),程序在运行过程中会根据需要(如传入的运算符参数)实例化其中的某个(尤其是这些实例化代码写在整个项目的多个不同地方时),我们可以写一个工厂函数,让它通过不同的参数去做具体的实例化调度。

这样做的好处有:
①避免程序中到处出现的if-else或switch-case分支结构;
②便于维护,每次增加或者删除都只需要修改工厂即可(这也是模块化编程,即函数的优点)。


开始操作

进入到敲代码的部分,我们先改服务端。首先,我的服务端中有三个Servlet:
Web APP文件结构

DispatcherServlet

这个Servlet就是我们的工厂,也是作为我们整个工程的唯一请求入口:

/*** Http 请求分派器
* 通过判断请求参数RequestType的值,将请求分派给具体的Servlet** @author Implementist*/
public class DispatcherServlet extends HttpServlet {@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {String RequestType = request.getParameter("RequestType");switch (RequestType) {case "Login":RequestDispatcher dispatchToLogin = request.getRequestDispatcher("LoginServlet");dispatchToLogin.forward(request, response);break;case "Register":RequestDispatcher dispatchToRegister = request.getRequestDispatcher("RegisterServlet");dispatchToRegister.forward(request, response);break;}}@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doPost(request, response);} }

实际上的功能就是,根据请求中传来的RequestType(请求类型)进行判断,然后将请求转发给相应的Servlet做真正的登录或者注册请求。

web.xml中的配置

在web.xml中,需要对三个Servlet做以下的注册和映射,注意DispatcherServlet的url-pattern

<web-app version="3.0" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"><servlet><servlet-name>DispatcherServletservlet-name><servlet-class>com.Implementist.myfirstapp.DispatcherServletservlet-class>servlet><servlet><servlet-name>LoginServletservlet-name><servlet-class>com.Implementist.myfirstapp.LoginServletservlet-class>servlet><servlet><servlet-name>RegisterServletservlet-name><servlet-class>com.Implementist.myfirstapp.RegisterServletservlet-class>servlet><servlet-mapping><servlet-name>DispatcherServletservlet-name><url-pattern>/*url-pattern>servlet-mapping><servlet-mapping><servlet-name>LoginServletservlet-name><url-pattern>/LoginServleturl-pattern>servlet-mapping><servlet-mapping><servlet-name>RegisterServletservlet-name><url-pattern>/RegisterServleturl-pattern>servlet-mapping><session-config><session-timeout>30session-timeout>session-config>
web-app>

客户端

①Volley里的url
现在,我们就可以给每一个Volley函数里写一样的URL了,注意最后是写服务端DispatcherServlet的映射符号‘ * ’

//域名形式
String url = "http://implementist.com/myfirstapp/*";//IP形式
String url = "http://1.1.1.1/myfirstapp/*";

②Volley函数的getParams函数
不能忘了,我们还需要给请求中填入RequestType请求类型,以使分派器知道该如何分派请求。

@Override
protected Map<String, String> getParams() throws AuthFailureError {Map<String, String> params = new HashMap<>();params.put("RequestType", "Login");...return params;
}

后记

这篇帖子很短,但是对于我们培养良好的代码习惯,写出易于维护的代码还是助益良多的。

最近学习了反射,发现上面DispatcherServlet的代码可以去除Switch-Case结构,使代码变得十分简洁,修改后的DispatcherServlet代码:

public class DispatcherServlet extends HttpServlet {@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {String RequestType = request.getParameter("RequestType");RequestDispatcher dispatcher = request.getRequestDispatcher(RequestType + "Servlet");dispatcher.forward(request, response);}@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {doPost(request, response);}
}


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章

立即
投稿

微信公众账号

微信扫一扫加关注

返回
顶部