Servlet的多线程单实例导致的数据同步问题

简介:

Servlet类本质上也是一个普通的类,并且Servlet容器默认只允许单个实例存在。当请求达到服务器的时候,Servlet实例如果已经存在的话则直接加载该实例,如果该Servlet类还未实例化则会先初始化这个Servlet。当请求到达Web服务器时,Web服务器中有一个线程池,它会从线程池中取一个工作线程,通过该线程调用请求的Servlet。因此,对Servlet来说,可以同时被好几个请求调用。请求结束后,线程放回线程池。

这种设计带来的好处是,Servlet单实例,减少了生成Servlet的开销。通过线程池响应请求,避免了不断创建线程和销毁线程的开销,提高了性能。但是这种多线程操纵单实例的模式,也会有一些副作用,那就是可能造成数据的不一致。看一个例子。

假设Servlet类如下:

 
  1. public class HelloServlet extends HttpServlet 
  2.     private String msg; 
  3.     @Override 
  4.     protected void doGet(HttpServletRequest req, HttpServletResponse resp) 
  5.     throws ServletException, IOException 
  6.     { 
  7.         msg = req.getParameter("msg"); 
  8.         try 
  9.         { 
  10.             Thread.sleep(10000); 
  11.         } 
  12.         catch(Exception ex) 
  13.         { 
  14.             ex.printStackTrace(); 
  15.         } 
  16.         resp.getWriter().println(msg); 
  17.     } 

读取参数msg,然后等待10s后打印出来。

但是当2个用户在调用这个类的时候,就会有冲突了。

看下图的url,A用户先输入参数为hello,1秒钟之后,B用户后输入world。

5B1D7EF5A6DC4A4E850D24F2EB15253D610E059935654786BC394818E7936E93

显示的都是world。

原因很简单,因为两个线程都调用同一个实例,A用户把成员变量msg设成hello后,B用户又将他改成了world。

因此会有这种结果出现。

解决方法主要有以下3种,

1,使用Javax.servlet.SingleThreadModel(Servlet2.4中已经废弃该接口),此时Servlet容器将保证Servlet实例以单线程方式运行,也就是说,同一时刻,只会有一个线程执行Servlet的service()方法。

2.去除实例变量,使用局部变量。

3.使用同步代码块:synchronized{…}

最常用的方法还是第2种,因为他使用了局部变量,互不冲突,相比刻意的单线程和加锁机制,效率更高。因此代码只要如下修改即可。

 
  1. public class HelloServlet extends HttpServlet 
  2.     //private String msg; 
  3.     @Override 
  4.     protected void doGet(HttpServletRequest req, HttpServletResponse resp) 
  5.     throws ServletException, IOException 
  6.     { 
  7.         String msg = req.getParameter("msg"); 
  8.         try 
  9.         { 
  10.             Thread.sleep(10000); 
  11.         } 
  12.         catch(Exception ex) 
  13.         { 
  14.             ex.printStackTrace(); 
  15.         } 
  16.         resp.getWriter().println(msg); 
  17.     } 













本文转自cnn23711151CTO博客,原文链接: http://blog.51cto.com/cnn237111/1154858 ,如需转载请自行联系原作者



相关文章
|
1月前
|
存储 编解码 算法
【ffmpeg音视频同步】解决ffmpeg音视频中多线程之间的数据同步问题
【ffmpeg音视频同步】解决ffmpeg音视频中多线程之间的数据同步问题
39 2
|
8月前
|
安全 Java 微服务
美团Java研发岗二面:什么叫线程安全?servlet是线程安全吗?
再过一个月的时间就是金九银十了,相信不少的朋友对于已经做好了充足的准备来应对这一次春招,小编对大家也没有太多的建议和帮助,分享一些你们面试上面可以用得到的一些面试题。今天主要给大家分享多线程(线程安全)方面的面试题,看到文章的小伙伴一定要看到后面哦,有海量的面试题分享给大家哟。
|
11月前
|
Java
Java:从单线程计数器到多线程数据同步synchronized和原子类Atomic
Java:从单线程计数器到多线程数据同步synchronized和原子类Atomic
118 0
|
11月前
|
安全 Java 容器
你能证明Servlet线程不安全吗?
你能证明Servlet线程不安全吗?
|
存储 安全
Servlet线程安全问题
Servlet线程安全问题(ThreadSafeServlet)
|
应用服务中间件
Servlet的线程不安全问题【使用局部变量是一种解决线程不安全的思路】
Servlet的线程不安全问题【使用局部变量是一种解决线程不安全的思路】
111 0
|
XML 安全 Java
漫画 | Servlet属于线程安全的吗?
漫画 | Servlet属于线程安全的吗?
116 0
漫画 | Servlet属于线程安全的吗?
|
安全 Java 容器
Servlet实例数量到底多少,是否线程安全
通过注解或servlet声明都能控制 servlet 容器如何提供 servlet 实例。
117 0
|
缓存 安全 Java
聊聊Servlet、Struts1、Struts2以及SpringMvc中的线程安全
前言 很多初学者,甚至是工作1-3年的小伙伴们都可能弄不明白?servlet Struts1 Struts2 springmvc 哪些是单例,哪些是多例,哪些是线程安全? 在谈这个话题之前,我们先了解一下Java中相关的变量类型以及内存模型JMM。
1934 0