1. Sevlet的生命周期(重点)
1.1 Servlet重要的四个生命周期方法
- 构造方法:
创建servlet对象的时候调用。默认情况下,第一次访问servlet的时候创建servlet对象只调用1次。证明servlet对象在tomcat是单实例的。 - init方法: 创建完servlet对象的时候调用。只调用1次。
- service方法: 每次发出请求时调用。调用n次。
- destroy方法: 销毁servlet对象的时候调用。停止服务器或者重新部署web应用时销毁servlet对象。只调用1次。
1.2 伪代码演示servlet的生命周期
Tomtcat内部代码运行:
1. 通过映射找到到servlet-class的内容,字符串: com.servlet.FirstServlet
2. 通过反射构造FirstServlet对象
2.1 得到字节码对象:
Class clazz = class.forName("com.servlet.FirstServlet");
2.2 调用无参数的构造方法来构造对象:
Object obj = clazz.newInstance(); //NO1:servlet的构造方法被调用
3. 创建ServletConfig对象,通过反射调用init方法
3.1 得到方法对象
Method m = clazz.getDeclareMethod("init",ServletConfig.class);
3.2 调用方法
m.invoke(obj,config);//NO.2 servlet的init方法被调用
4. 创建request,response对象,通过反射调用service方法
4.1 得到方法对象
Methodm m =clazz.getDeclareMethod("service",HttpServletRequest.class,HttpServletResponse.class);
4.2 调用方法
m.invoke(obj,request,response);//NO3. servlet的service方法被调用
5. 当tomcat服务器停止或web应用重新部署,通过反射调用destroy方法
5.1 得到方法对象
Method m = clazz.getDeclareMethod("destroy",null);
5.2 调用方法
m.invoke(obj,null); //NO4.servlet的destroy方法被调用
1.3 用时序图来演示servlet的生命周期

2. Servlet的多线程并发问题
首先需要注意的是servlet对象在tomcat服务器是单实例多线程的。
所以当多个servlet的线程同时访问了servlet的共享数据,如成员变量,可能会引发线程安全问题。
解决办法:
- 把使用到共享数据的代码块进行同步(使用synchronized关键字进行同步)
- 建议在servlet类中尽量不要使用成员变量。如果确实要使用成员,必须同步。而且尽量缩小同步代码块的范围。(哪里使用到了成员变量,就同步哪里!!),以避免因为同步而导致并发效率降低。
线程安全代码:
package com.servlet;
import java.io.IOException;
import java.util.Date;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class ServetlDemo4 extends HttpServlet {
private int i = 1;
@Override
public void init() throws ServletException {
System.out.println("ServetlDemo4...init()");
}
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 设置编码格式
// resp.setContentType("text/html;charset=utf-8");
resp.setCharacterEncoding("utf-8");// 内容编码,防止出现中文乱码
resp.setContentType("text/html;charset=utf-8");
synchronized (ServetlDemo4.class) {
// 向浏览器输出内容
resp.getWriter().write("这是第" + i + "次访问...");
try {
Thread.sleep(5000);
} catch (Exception e) {
// TODO: handle exception
}
i++;
}
}
@Override
public void destroy() {
System.out.println("ServetlDemo4...destroy()");
}
}
此时通过两个浏览器窗口进行访问就会出现两个“第一次访问”的线程安全问题
本文详细解析Servlet的生命周期,包括构造方法、init、service和destroy的调用时机。并通过示例代码展示如何解决Servlet多线程并发访问时的线程安全问题。
930

被折叠的 条评论
为什么被折叠?



