Servlet的多线程问题

来源:互联网 发布:彩影软件免费下载 编辑:程序博客网 时间:2024/05/10 17:28

        我们都知道Servlet是默认的以多线程进行执行的,一般情况下一个Servlet只会存在一个实例,Servlet容器在多个客户端请求该Servlet的的情况下会开启多个线程同时访问这一个Servlet实例。这样,当两个或多个线程同时访问同一个Servlet时,可能会发生多个线程同时访问同一个资源的情况,这时,数据就可能变得不一致了。

下面以一个例子来说明:

结果如下所示:

        这种情况肯定不是我们想看到的,而且也是不允许出现的情况。

        下面我简单的介绍一下解决该问题的几种方式,并进行一个对比分析。

1)通过继承SingleThreadModel接

      

2)使用synchronized同步共享数据


此处看过某些人写的,把username = req.getParameter("username");写在了synchronized(this)的外面,其实这样是无法锁住的,经过本人亲测也证实了这个想法。

3)使用局部变量代替成员变量

三种解决方法的优劣分析:

1)如果一个Servlet实现了SingleThreadModel接口,Servlet容器将为每个新的请求创建一个单独的Servlet实例,这将引起大量的系统开销。而且通过那道横线(SingleThreadModel)也可以看出在Servlet 2.4中是不建议使用这种方式的。

2)如果在程序中使用同步来保护要使用的共享的数据,也会使系统的性能大大下降。这是因为被同步的代码块在同一时刻只能有一个线程执行它,使得其同时处理客户请求的吞吐量降低,而且很多客户处于阻塞状态。另外为保证主存内容和线程的工作内存中的数据的一致性,要频繁地刷新缓存,这也会大大地影响系统的性能。所以在实际的开发中也应避免或最小化 Servlet 中的同步代码。

3)在Servlet中尽量使用局部变量是解决Servlet线程安全的最佳选择。从Java的内存模型也可以知道,方法中的局部变量是在栈上分配空间,而且每个线程都有自己私有的栈空间,所以它们不会影响线程的安全。