Servlet请求分派器——DispatcherServlet

  • Post author:
  • Post category:其他





前提



索引文章

中我提到过,我会写一篇关于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 请求分派器<br>
 * 通过判断请求参数RequestType的值,将请求分派给具体的Servlet
 *
 * @author Implementist
 */
public class DispatcherServlet extends HttpServlet {
    @Override
    protected 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;
        }
    }

    @Override
    protected 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>DispatcherServlet</servlet-name>
        <servlet-class>com.Implementist.myfirstapp.DispatcherServlet</servlet-class>
    </servlet>
    <servlet>
        <servlet-name>LoginServlet</servlet-name>
        <servlet-class>com.Implementist.myfirstapp.LoginServlet</servlet-class>
    </servlet>
    <servlet>
        <servlet-name>RegisterServlet</servlet-name>
        <servlet-class>com.Implementist.myfirstapp.RegisterServlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>DispatcherServlet</servlet-name>
        <url-pattern>/*</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>LoginServlet</servlet-name>
        <url-pattern>/LoginServlet</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>RegisterServlet</servlet-name>
        <url-pattern>/RegisterServlet</url-pattern>
    </servlet-mapping>
    <session-config>
        <session-timeout>
            30
        </session-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 {
    @Override
    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        String RequestType = request.getParameter("RequestType");
        RequestDispatcher dispatcher = request.getRequestDispatcher(RequestType + "Servlet");
        dispatcher.forward(request, response);
    }

    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        doPost(request, response);
    }
}



版权声明:本文为Mr_Megamind原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。