理解两个问题足以,问题如下:
1、如果不同的2个用户同时对这个网站的不同业务同时发出请求(如注册和登陆),那容器里有几个servlet呢??
2、不同的用户同时对同一个业务(如注册)发出请求,那这个时候容器里产生的有是几个servlet实例呢?
答案:
引子:一个web容器,可以有多个servlet。 对提交到同一个servlet类的多个业务请求,共享一个servlet对象(即这个servlet类只被实例化一次)
但别忘了,请求还可以从一个servlet forward到另一个servlet,因此一个请求是可以产生多个servlet的,但是由不同的servlet类实例化的,每个servlet类都只被实例化一次,直到应用程序终止或服务器shutdown
问题1的答案:容器里有2个servlet(当然,这是在“一个servlet对应一种业务请求”的前提下,如果你要把两个业务逻辑写在同一个servlet内另当别论了)
问题2的答案:只有一个servlet实例。一个servlet是在第一次被访问时加载到内存并实例化的。同样的业务请求共享一个servlet实例。不同的业务请求一般对应不同的servlet. 想也知道拉,如果一个网站要被几千万人同时登录,如果创建几千万个实例的话服务器还怎么跑得动?
一、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 中创建它们各自的变量。
因此,方法中的变量不存在线程不安全问题。
public class ExampleServlet extends HttpServlet {
private Object thisIsNOTThreadSafe;
protected void doGet(HttpServletRequest request,
HttpServletResponse response) throws ServletException, IOException {
// BAD!! Shared among all requests!
thisIsNOTThreadSafe = request.getParameter("foo");
// OK, this is thread safe.
Object threadSafeObj;
threadSafeObj = request.getParameter("foo");
}
}
四、拓展:Struts2 中的 Action 对象是单例吗?
不是。
当请求到来时,Web 容器为每一个请求创建一个 Request 和 Response 对象。
然后再创建一个线程,并把这两个对象的引用指向该线程。
Struts2 中的 Action 对象是 re-created 的,为每一个请求。并绑定到 Request 对象上,
作为 Request 对象的一个属性。这样就不存在线程不安全问题,因为每一个 Request 对象
只绑定一个线程。
注意:不要混淆 Struts DispatcherFilter 和 StrutsAction
转自网络