Servet

时间:2023-03-08 23:40:05
Servet

一、Servlet 是单例吗

不是。 
1、你可以用多个 URL 映射同一个 Servlet。这样就会出现多个实例。

2、看看 Servlet 定义:

引用
For a servlet not hosted in a distributed environment (the default), the servlet container must use only one instance per servlet declaration. 
如果 servlet 不是在分布式环境下(默认),servlet 容器必须使一个 servlet 实例对应一个 servlet 声明。

However, for a servlet implementing the SingleThreadModel(Deprecated) interface, the servlet container may instantiate multiple instances to handle a heavy request load and serialize requests to a particular instance. 
然而,实现了 SingleThreadModel 接口的 Servlet,可以有多个实例。以处理繁重的请求,并且序列化 request 到特定的 servlet 实例。

public interface SingleThreadModel 
Ensures that servlets handle only one request at a time. 

结论: 
虽然 Servlet 在多数情况下只有一个实例。但它并不是单例设计模式,即不是真正的单例。

二、Servlet 为什么是线程不安全的

基于 JVM 对多线程的支持,这样可以提高代码的执行效率。 
不需要为每一个请求都要单独创建/销毁 Servlet(执行 init(), desdroy() )。 
同一段代码可以在同一时间被多个请求同时执行。

Servlet 是普通的 Java 类,因此没有对其做线程安全的处理。

三、如何保证线程安全?(避免不安全)

但是, 
Java 的类是线程安全的,只有在一种情况下:该类没有 instance variables. 
即,没有(实例)变量时。

实例变量(instance properties)是声明在类中的变量,而不是声明在方法中的变量。

声明在方法中的变量是线程安全的,因为在执行该方法时,每一个线程都会在 Stack 中创建它们各自的变量。 
因此,方法中的变量不存在线程不安全问题。

  1. public class ExampleServlet extends HttpServlet {
  2. private Object thisIsNOTThreadSafe;
  3. protected void doGet(HttpServletRequest request,
  4. HttpServletResponse response) throws ServletException, IOException {
  5. // BAD!! Shared among all requests!
  6. thisIsNOTThreadSafe = request.getParameter("foo");
  7. // OK, this is thread safe.
  8. Object threadSafeObj;
  9. threadSafeObj = request.getParameter("foo");
  10. }
  11. }

四、拓展:Struts2 中的 Action 对象是单例吗?

不是。 
当请求到来时,Web 容器为每一个请求创建一个 Request 和 Response 对象。 
然后再创建一个线程,并把这两个对象的引用指向该线程。

Struts2 中的 Action 对象是 re-created 的,为每一个请求。并绑定到 Request 对象上, 
作为 Request 对象的一个属性。这样就不存在线程不安全问题,因为每一个 Request 对象 
只绑定一个线程。

注意:不要混淆 Struts DispatcherFilter 和 StrutsAction