详细分析Sevlet的生命周期与线程安全问题

本文详细解析Servlet的生命周期,包括构造方法、init、service和destroy的调用时机。并通过示例代码展示如何解决Servlet多线程并发访问时的线程安全问题。

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的共享数据,如成员变量,可能会引发线程安全问题。

解决办法:

  1. 把使用到共享数据的代码块进行同步(使用synchronized关键字进行同步)
  2. 建议在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()");

	}

}

此时通过两个浏览器窗口进行访问就会出现两个“第一次访问”的线程安全问题

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值