前提
在
索引文章
中我提到过,我会写一篇关于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:
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);
}
}