//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方法
// ....
}
}
package com.bjpowernode.javaweb.servlet;
import javax.servlet.*;
import java.io.IOException;
/**
* 编写一个标准通用的Servlet,起名:GenericServlet
* 以后所有的Servlet类都不要直接实现Servlet接口了。
* 以后所有的Servlet类都要继承GenericServlet类。
* GenericServlet 就是一个适配器。
*/
public abstract class GenericServlet implements Servlet {
// 成员变量
private ServletConfig config;
/**
* init方法中的ServletConfig对象是小猫咪创建好的。
* 这个ServletConfig对象目前在init方法的参数上,属于局部变量。
* 那么ServletConfig对象肯定以后要在service方法中使用,怎么才能保证ServletConfig对象在service方法中能够使用呢?
*/
@Override
public void init(ServletConfig config) throws ServletException {
//System.out.println("servletConfig对象,小猫咪创建好的:" + config);
this.config = config;
// 调用init()方法
this.init();
}
/**
* 这个init方法是供子类重写的。
*/
public void init(){
System.out.println("GenericServlet 重写的init方法执行");
}
@Override
public ServletConfig getServletConfig() {
return config;
}
/**
* 抽象方法,这个方法最常用。所以要求子类必须实现service方法。
* @param servletRequest
* @param servletResponse
* @throws ServletException
* @throws IOException
*/
public abstract void service(ServletRequest servletRequest, ServletResponse servletResponse)
throws ServletException, IOException;
@Override
public String getServletInfo() {
return null;
}
@Override
public void destroy() {
}
}
package com.bjpowernode.javaweb.servlet;
import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import java.io.IOException;
public class LoginServlet extends GenericServlet{
// 思考一个问题:
// 有没有一种可能,需要我在LoginServlet类中重写init方法?
// 当然有可能,于是乎就重写了init方法。
/*public void init(ServletConfig config) throws ServletException {
System.out.println("重写我自己的规则。。。。。");
}*/
// 父类将原始的init方法final了,我子类没有办法重写这个init方法了。
// 如果这个时候我还是希望能够重写init方法,该怎么办呢?
@Override
public void init(){
System.out.println("LoginServlet's init() method execute!");
}
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("正在处理用户登录请求,请稍后。。。。。");
// 想在LoginServlet子类中使用ServletConfig对象怎么办?
ServletConfig config = this.getServletConfig();
System.out.println("service方法中是否可以获取到ServletConfig对象?" + config);
}
}
用户向浏览器发送请求后,服务器的执行流程?
① 服务器解析用户的请求路径,然后通过反射创建LoginServlet对象,此时会调用无参构造。
(这里无参构造具体是先父类构造方法再子类构造方法吗???)
② 然后Tomcat源码写了调用含servletConfig参数的init函数,此时虽然是调用的servlet接口的含servletConfig参数的init函数,实际上调用对应实现类 LoginServlet的含servletConfig参数的init函数
(这里不知道servlet接口对应实现类是 LoginServlet还是GenericServlet,没猜错的话servlet接口对应实现类就是 LoginServlet,因为经调试看出,后面的this.init( )的this指向了LoginServlet???)
③ 因为对应实现类 LoginServlet没有含servletConfig参数的init函数,所以在LoginServlet的父类GenericServlet 里面找到含servletConfig参数的init函数
④ GenericServlet 里面找到含servletConfig参数的init函数里面此时的this指的是 LoginServlet,this.init( )表示调用 LoginServlet里面的无参的init函数
⑤ 然后调用LoginServlet的service方法
为什么一定要另外写一个允许子类重写的init空参函数,为什么不直接在子类LoginServlet直接重写父类的含ServletConfig的init函数,里面写上super.init(config)来将父类的config赋值。???
public void init(ServletConfig config) throws ServletException {
super.init(config);//调用父类的init函数
System.out.println("LoginServlet's init(ServletConfig config) method execute!");
}