一、servlet产生多线程安全问题的原因
1.多线程问题产生的原因
多线程的产生主要由于以下几点,必须同时成立,才会产生多线程问题。
- 首先要有多步操作
- 还要有多个线程同时访问
- 最后还要有共享的资源
2.servlet产生多线程问题的原因
- servlet所做的处理一般都不是一步就能完成的
- 首先从servlet对象的产生说起,servlet是用来服务多线程的,多个客户可以同时访问同一个servlet对象
- servlet是单例的,只有一个对象,所以它的成员变量是被访问者共享的
综上servlet存在线程安全问题,下面我们通过一段代码来演示一下servlet的多线程问题。
package com.csdn.servlet;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.SingleThreadModel;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class ThreadServletDemo5 extends HttpServlet {
private int i=1;
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
response.setContentType("text/html;charset=utf-8");
System.out.println("您是第"+i+"位访客");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
i++;
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doGet(request, response);
}
}
最后的输出结果如下图所示
分析代码的实现过程,代码原来的作用是为了记录访问量,然而实际上执行的却没达到我们预期的效果,原因就在于该方案的实现中出现了多线程问题。
二、解决方案
1.可以让servlet对象实现Single ThreadModel接口,servlet的产生本身就是服务多线程的,而实现该接口后,servlet对象就成为了单线程的,所以该方案不予采用。
2.使用synchronized 关键字能保证一次只有一个线程可以访问被保护的区段,Servlet可以通过同步块操作来保证线程的安全。
3.我们必须尽量避免在servlet中使用成员变量,因为servlet是默认单例运行,可以采用局部变量,成员变量的使用将会产生线程安全问题。
此例中,我们必须采用成员变量来记录访问量,此时,我们可以采用第二种方案,方案的代码如下所示:
package com.csdn.servlet;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.SingleThreadModel;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class ThreadServletDemo5 extends HttpServlet {
private int i=1;
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
synchronized(this){
response.setContentType("text/html;charset=utf-8");
System.out.println("您是第"+i+"位访客");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
i++;
}
}
public void doPost(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
this.doGet(request, response);
}
}
最后得到的结果如下图所示: