JavaWeb|适配器(GenericServlet)改造Servlet

JavaWeb系列教程
JavaWeb—Servlet
模拟Servlet本质
使用IDEA开发Servlet程序
Servlet对象的生命周期
适配器(GenericServlet)改造Servlet
ServletConfig
Servlet–ServletContext
HttpServlet源码分析
web站点欢迎页
一篇学会HttpServletRequest
如果大家觉得有帮助的话,不妨动动小手,点赞收藏一波,也方便后面的复习哈
JavaWeb|适配器(GenericServlet)改造Servlet
文章图片

我们发现在编写一个Servlet类的时候,要实现它的所有方法,但是事实上,我们一般只是使用service方法,这样就会让代码看着很丑陋,那么我们要怎么解决呢?
我们在说解决问题之前,先来看看这样的一个例子
适配器设计模式Adapter
  • 手机直接插到220V的电压上,手机直接就报废了。怎么办?可以找一个充电器。这个充电器就是一个适配器。手机连接适配器。适配器连接220V的电压。这样问题就解决了。
    同样的道理,我们也可以使用这样的思想,来解决这个问题
  • 编写一个GenericServlet类,这个类是一个抽象类,其中有一个抽象方法service。
    GenericServlet实现Servlet接口。
    GenericServlet是一个适配器。
    以后编写的所有Servlet类继承GenericServlet,重写service方法即可。
package com.bjpowernode.servlet; import jakarta.servlet.*; import java.io.IOException; import java.io.PrintWriter; /** * @author zengyihong * @create 2022--03--06 20:08 */ public class AServlet extends GenericServlet { @Override public void service(ServletRequest request, ServletResponse response) throws ServletException, IOException {response.setContentType("text/html"); PrintWriter out = response.getWriter(); //获取ServletContext对象 ServletConfig config = this.getServletConfig(); ServletContext context = config.getServletContext(); out.print(context); } }

思考:GenericServlet类是否需要改造一下?怎么改造?更利于子类程序的编写?
  • 思考第一个问题:我们提供了一个GenericServlet之后,init方法还会执行吗?
    • 还会执行。会执行GenericServlet类中的init方法。
    • 很明显,子类中如果这个方法,它就会去父类找相应的方法
  • 【JavaWeb|适配器(GenericServlet)改造Servlet】思考第二个问题:init方法是谁调用的?
    • Tomcat服务器调用的。
  • 思考第三个问题:init方法中的ServletConfig对象是谁创建的?是谁传过来的?
    • 都是Tomcat干的。
    • Tomcat服务器先创建了ServletConfig对象,然后调用init方法,将ServletConfig对象传给了init方法。
    • 如果对上面几个问题不大理解的话,大家可以先去看看servlet的生命周期
  • 思考一下Tomcat服务器伪代码:
public class Tomcat { public static void main(String[] args){ // ..... // Tomcat服务器伪代码 // 创建LoginServlet对象(通过反射机制,调用无参数构造方法来实例化LoginServlet对象) Class clazz = Class.forName("com.bjpowernode.javaweb.servlet.LoginServlet"); Object obj = clazz.newInstance(); // 向下转型 Servlet servlet = (Servlet)obj; // 创建ServletConfig对象 // Tomcat服务器负责将ServletConfig对象实例化出来。 // 多态(Tomcat服务器完全实现了Servlet规范) ServletConfig servletConfig = new org.apache.catalina.core.StandardWrapperFacade(); // 调用Servlet的init方法 servlet.init(servletConfig); // 调用Servlet的service方法 // ....} }

但是ServletConfig对象以后肯定是要在service方法中使用的,那么怎么才可以保证ServletConfig对象在service方法中能够使用呢? 我们很容易就想到了成员变量
init方法中的ServletConfig对是Tomcat服务器创建好的
  • 这个ServletConfig对象目前在init方法的参数上,属于局部变量
  • 那么怎么才能使用呢?------成员变量
  • ServletConfig对象将来肯定是要用在service方法的,怎么保证ServletConfig对象在service方法中调用呢-----调用getServletConfig()方法
思考一下,还有一种可能,需要子类重写父类的init方法,但是这样就会导致父类的init方法不执行,那么ServletConfig对象就是null了,那么将来就会有很大的麻烦,我们就不要让这个方法被子类重写,可以在方法加上final修饰,但是如果子类就是想要重写呢? 但是这个时候子类没有办法重写,那到底要怎么办呢
解决办法:再GenericServlet类中添加一个无参的init方法,供子类进行重写。
JavaWeb|适配器(GenericServlet)改造Servlet
文章图片

上面那个过程都是我们自己一步一步根据需求推出来的,是不是感觉好麻烦,一会这样,一会那样的,不过好消息是,GenericServlet这个类不需要我们写,官方已经提供好了,而且官方写的代码考虑的肯定比我们周全

// // Source code recreated from a .class file by IntelliJ IDEA // (powered by FernFlower decompiler) //package jakarta.servlet; import java.io.IOException; import java.io.Serializable; import java.util.Enumeration; public abstract class GenericServlet implements Servlet, ServletConfig, Serializable { private static final long serialVersionUID = 1L; private transient ServletConfig config; public GenericServlet() { }public void destroy() { }public String getInitParameter(String name) { return this.getServletConfig().getInitParameter(name); }public Enumeration> getInitParameterNames() { return this.getServletConfig().getInitParameterNames(); }public ServletConfig getServletConfig() { return this.config; }public ServletContext getServletContext() { return this.getServletConfig().getServletContext(); }public String getServletInfo() { return ""; }public void init(ServletConfig config) throws ServletException { this.config = config; this.init(); }public void init() throws ServletException { }public void log(String message) { this.getServletContext().log(this.getServletName() + ": " + message); }public void log(String message, Throwable t) { this.getServletContext().log(this.getServletName() + ": " + message, t); }public abstract void service(ServletRequest var1, ServletResponse var2) throws ServletException, IOException; public String getServletName() { return this.config.getServletName(); } }

大家看到我们所写和官网提供的思路其实是大同小异的。
希望得到大家的支持JavaWeb|适配器(GenericServlet)改造Servlet
文章图片

    推荐阅读