当多个客户端并发访问同一个servlet时,web服务器会为每一个客户端的访问请求创建一个线程,并在这个线程上调用Servlet的service方法,因为service方法内如果访问了同一资源的话,就有可能引发线程安全问题。
如果某个servlet实现了SingleThreadModel接口,那么servlet引擎将以单线程模式来调用其service方法。
在servlet中测试线程安全问题的发生:
package cn.sun;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
//线程安全
@WebServlet("/servlet/ServletDemo4")
public class ServletDemo4 extends HttpServlet {
int i=0; //操作同一共享资源
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
i++;
try {
Thread.sleep(1000*10); //线程休眠
} catch (InterruptedException e) {
e.printStackTrace();
}
response.getOutputStream().write((i+"").getBytes());
}
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
}
}
运行http://localhost:8080/day05/servlet/ServletDemo4:(结果本应为1)
发生了线程安全问题。
这是我们可以将操作同一资源的代码放在同步代码块synchronize中,这样就可以解决线程安全问题。
public class ServletDemo4 extends HttpServlet {
int i=0; //操作同一共享资源
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
synchronized (this) { //同步代码块重点内容
i++;
try {
Thread.sleep(1000*10); //线程休眠
} catch (InterruptedException e) {
e.printStackTrace();
}
response.getOutputStream().write((i+"").getBytes());
}
}
}
运行结果http://localhost:8080/day05/servlet/ServletDemo4:
**
但是
,这样的方法在web应用中却不能使用,这样会导致一个用户在访问web网页时,另外的用户不能访问。**
在servlet中,为了解决这一问题,可以让这个servlet去实现SingleThreadModel接口(已过时)servlet引擎将以单线程模式来调用其service方法。
SingleThreadModel接口中没有定义任何方法(标记接口)。
SingleThreadModel接口的实现方式是产生多个servlet实例对象,并发的每个线程分别调用一个独立的servlet实例对象,这样servlet引擎会创建多个servlet实例对象。但是解决多线程的安全问题是指一个servlet实例对象被多个线程同时调用的问题,所以SingleThreadModel接口已过时。
package cn.sun;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.SingleThreadModel;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
//线程安全
@WebServlet("/servlet/ServletDemo4")
public class ServletDemo4 extends HttpServlet implements SingleThreadModel{ //只需增加实现SingleThreadModel接口的声明即可
int i=0; //操作同一共享资源
//子类在覆盖父类的方法时,不能抛出比父类更多的异常
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
i++;
try {
Thread.sleep(1000*10); //线程休眠
} catch (InterruptedException e) {
e.printStackTrace();
}
response.getOutputStream().write((i+"").getBytes());
}
}