转载地址:https://www.cnblogs.com/LipeiNet/p/5699944.html

前言:前面说了很多关于Servlet的一些基础知识,这一篇主要说一下关于Servlet的线程安全问题。

1:多线程的Servlet模型

要想弄清Servlet线程安全我们必须先要明白Servlet实例是如何创建,它的模式是什么样的。

在默认的情况下Servlet容器对声明的Servlet,只创建一个Servlet实例,那么如果要是多个客户同时请求访问这个Servlet,Servlet容器就采取多线程。下面我们来看一幅图

从图中可以看出当客户发送请求的时候,Servlet容器通过调度者线程从线程池中选择一个线程,然后将请求传递给这个线程,然后在由这个线程去执行Servlet的Service方法。

如果多个客户端同时请求执行一个Servlet实例,那么这个Servlet容器的Service方法将在多个线程中并发执行(比喻图中客户1,客户2,客户3同时调用Servlet1实例,那么调度者线程就会在线程池中调用3个线程分别用于客户1,2,3的请求,然后3个线程同时并发执行Servlet1实例的Service方法)因为Servlet容器采取的单实例多线程的方法,那么就大大的减小了Servlet实例创建的开销,提升了对请求的响应时间,也是这样引起了Servlet线程安全问题。所以我们下面说线程安全问题。

2:Servlet的线程安全

2.1:变量的线程安全

2.1.1:变量为啥会存在线程安全

我们先看一段代码

 1 public class HelloWorldServlet extends HttpServlet{
2 private String userName;
3 protected void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException
4 {
5 userName=request.getParameter("userName");
6 PrintWriter out=response.getWriter();
7 if(userName!=null&&userName!="")
8 {
9 out.print(userName);
10 }
11 else {
12 out.println("用户名不存在");
13 }
14 }
15 }

我们来分析这段代码,现在有A,B2个客户端同时请求HelloWorldServlet 这个实例,Servlet容器分配线程T1来服务A客户端的请求,T2来服务B客户端的请求,操作系统首先调用T1来运行,T1运行到第6行的时候得到了用户名为张三并保存,此时时间片段到了,操作系统开始调用T2运行也运行到第6行但是这个用户名是李四,此时时间片段又到了,操作系统又开始运行T1,从第7行开始运行,但是此时的用户名却成了李四,输出的时候确实李四(很明显是错误的),那么这个时候就出现了线程安全问题。

2.1.2:如何防止变量的线程安全

  1. 把全局变量改为局部变量
    1. public class myservlet2 extends HttpServlet{

      int num=1;
      //servlet->GenericServlet->HttpServlet
      @Override
      public void init() throws ServletException {
      // TODO Auto-generated method stub
      super.init();
      System.out.println("HttpServlet...init...初始化");
      }

      @Override
      public void service(ServletRequest arg0, ServletResponse arg1)
      throws ServletException, IOException {
      // TODO Auto-generated method stub
      num++;
      System.out.println(num);
      System.out.println(arg0.getRemoteAddr());
      System.out.println("HttpServlet...servlet demo2");
      }

      @Override
      public void destroy() {
      // TODO Auto-generated method stub
      super.destroy();
      System.out.println("HttpServlet...destory...结束");
      }

      }

      因为每次调用这个方法的时候就会重写对userName实例化这样一来就不会存在线程安全问题了。

  2. 使用synchronized对doGet进行同步

protected synchronized  void doGet(HttpServletRequest request, HttpServletResponse response)

采用这种方式明显不合适,因为这样T2必须要等T1执行完毕以后才可以执行,大大的影响了效率。

3.如果是静态资源则加上final表示这个资源不可以改变

比喻 final static String url="jdbc:mysql://localhost:3306/blog";

2.2:属性的线程安全

在Servlet中可以访问保存在ServletContext,HttpSession,ServletRequest对象中的属性,这三种对象都提供了getAttribute(),setAttribute() 方法用来对取和设置属性,那么这三个不同范围对象的属性访问是否线程安全呢,下面我们来一起看一下

2.2.1:ServletContext

首先明确一点是ServletContext是被应用程序下所有的Servlet所共享的,那么ServletContext对象就可以被web应用程序所有的Servlet访问,那么这样一来多个Servlet就可以同时对ServletContext的属性进行设置和访问,所以这个时候就会出现线程安全问题。我们来看一段代码

 1 protected void service(HttpServletRequest request, HttpServletResponse response)
2 {
3 String userName=request.getParameter("userName");
4 if ("login") {
5 List list=(List)getServletContext().getAttribute("userList");
6 list.add(userName);
7 }
8 else {
9 List list=(List)getServletContext().getAttribute("userList");
10 list.remove(userName);
11 }
12 }
 1 protected void service(HttpServletRequest request, HttpServletResponse response) throws IOException
2 {
3 List list=(List)getServletContext().getAttribute("userList");
4 int count=list.size();
5 for(int i=0;i<count;i++)
6 {
7 PrintWriter out=response.getWriter();
8 out.println(list.get(i));
9 }
10 }

第一段代码是当用户登录以后把用户名保存在ServletContext属性中,如果不是登录就删除这个用户

第二段代码就是查看应用程序所有的用户登录情况,那么我们看如何出现线程安全问题的

当2个请求并发执行的时候,可能第二段代码刚刚执行第五行的时候获取的count=5;但是呢另一个请求恰好执行第一段代码第十行,把其中的某个用户删除了,当第二段代码在循环遍历的时候运行到count=5的时候就会数组超过索性界限异常。那么此时就出现了线程安全问题。那么遇到这样的问题怎么解决呢,第一就是把ServletContext属性值进行拷贝保存起来,第二就是采用synchronized 进行同步(这个效率低)

2.2.2:HttpSession

httpSession对象在用户会话期间存活的,不像ServletContext一样被所有的用户共享,所以说一个HttpSession在同一个时刻只用一个用户进行请求的,因此理论看来Session是线程安全的,其实并不是如此,这个和浏览器有关,在上一篇Session我们说过,同一个浏览器只能具有一个Session,那么这样一来就会出现Session线程安全问题,看如下代码

 1     protected void service(HttpServletRequest request, HttpServletResponse response) throws IOException
2 {
3 String commandType=request.getParameter("commandType");
4 HttpSession session=request.getSession();
5 List list=(List)session.getAttribute("items");
6 if ("add".equals(commandType)) {
7 //添加
8 }
9 else if("delete".equals(commandType)){
10 //删除
11 }
12 else {
13 int count=list.size();
14 for (int i = 0; i < count; i++) {
15 //遍历
16 }
17 }
18 }

上面是一个添加物品信息的一个简单伪代码,如果用户现在在一个浏览器窗口删除一件物品的同时又在另一个窗口去获取所有的物品这个时候就会出现线程安全,从上面的介绍得知Servlet容器是多线程单实例的,这个时候Servlet容器就会分配2个线程来分别为删除物品和获取所有物品进行服务,如果其中一个线程刚好运行到14行时间片段结束,另一个线程这个时候又运行第10行删除一条物品信息,然后第一个线程又开始运行第15开始遍历,此时同样出现了上面数组索性超出范围的错误。

2.2.3:HttpRequest

httprequest是线程安全的,因为每个请求都会调用Service,都会创建一个新的HttpRequest和局部变量一样。

3:SingleThreadModel

从名字很好理解,就是单线程模式,也就是说如果Servlet实现了SingleThreadModel接口,Servlet容器就保证一个时刻只有一个线程在Servlet实例的Service方法运行(其实和同步差不多)这样一来就很影响效率了,现在SingleThreadModel已经被废弃了,值得注意的是就算Servlet实现了SingleThreadModel接口并不一定保证线程安全,比喻上面说的ServletContext,HttpSession,因为ServletContext是应用程序共享的,可能2个Servlet实例同时运行造成线程安全,HttpSession因为是在同一浏览器共享的所以也会出现(虽然可能性很小)

4:总结

1:只要我们了解Servlet容器工作的模式,可能就能够理解为什么Servlet会出现线程安全问题,所以一定牢记Servlet容器是多线程单实例的模型

2:避免使用全局变量,最好是使用局部变量,其实这本身也是一个好的编程习惯

3:应该使用只读的实例变量和静态变量(就是前面加上final意为不可改变)

4:不要在Servlet上自己创建线程,因为Servlet容器已经帮我们做好了。

5:如果要修改共享对象的时候记得要同步,尽量缩小同步的范围(比喻修改Session时候直接使用synchronized(Session)即可),避免影响性能

Servlet线程安全问题(转载)的更多相关文章

  1. javaweb回顾第六篇谈一谈Servlet线程安全问题

    前言:前面说了很多关于Servlet的一些基础知识,这一篇主要说一下关于Servlet的线程安全问题. 1:多线程的Servlet模型 要想弄清Servlet线程安全我们必须先要明白Servlet实例 ...

  2. JAVAEE_Servlet_18_关于Servlet线程安全问题

    关于Servlet线程安全问题 Servlet线程安全 Servlet 是单实例多线程的环境下运行的. 在服务器运行期间,一个Servlet接口实现类,只能创建一个实例对象(一个进程(Servlet接 ...

  3. 开玩笑Web它servlet(五岁以下儿童)---- 如何解决servlet线程安全问题

    servlet默认值是安全线的存在,但说白,servlet安全线实际上是一个多线程线程安全问题.因为servlet它正好是一个多线程的安全问题出现. 每次通过浏览器http同意提交请求,将一个实例se ...

  4. Servlet线程安全问题

    Servlet采用单实例多线程方式运行,因此是线程不安全的.默认情况下,非分布式系统,Servlet容器只会维护一个Servlet的实例,当多个请求到达同一个Servlet时,Servlet容器会启动 ...

  5. (2.1)servlet线程安全问题

    本文参考链接:http://www.yesky.com/334/1951334.shtml 摘 要:介绍了Servlet多线程机制,通过一个实例并结合Java 的内存模型说明引起Servlet线程不安 ...

  6. IT兄弟连 JavaWeb教程 Servlet线程安全问题

    在Internet中,一个Web应用可能被来自西面八方的客户并发访问(即同时访问),而且有可能这些客户并发访问的是Web应用中的同一个Servlet,Servlet容器为了保证能同时相应多个客户端要求 ...

  7. javaweb学习总结二十三(servlet开发之线程安全问题)

    一:servlet线程安全问题发生的条件 如果多个客户端访问同一个servlet时,发生线程安全问题,那么它们访问的是相同的资源.如果访问 的不是相同资源,则不存在线程安全问题. 实例1:不会产生线程 ...

  8. JavaWeb学习之Servlet(三)----Servlet的映射匹配问题、线程安全问题

    [声明] 欢迎转载,但请保留文章原始出处→_→ 文章来源:http://www.cnblogs.com/smyhvae/p/4140529.html 一.Servlet映射匹配问题: 在第一篇文章中的 ...

  9. (转)JavaWeb学习之Servlet(三)----Servlet的映射匹配问题、线程安全问题

    [声明] 欢迎转载,但请保留文章原始出处→_→ 文章来源:http://www.cnblogs.com/smyhvae/p/4140529.html 一.Servlet映射匹配问题: 在第一篇文章中的 ...

随机推荐

  1. 题解【洛谷P1886】滑动窗口 /【模板】单调队列

    题面 单调队列模板题. 单调队列可以从队首和队尾出队. 队列中的元素大小具有一定的顺序. 具体可参考这一篇题解 #include <bits/stdc++.h> #define itn i ...

  2. PP: Data-driven classification of residential energy consumption patterns by means of functional connectivity networks

    Purpose Implement a good user aggregation and classification. or to assess the interrelation pattern ...

  3. SpringMVC请求乱码问题

    今天做一个项目实现插入数据的功能,最开始没有添加FilterEncoding处理字符集乱码了,那是正常的,后来我添加过之后依然还是乱码,让我 百思不得其解,代码配置如下: EncodingFilter ...

  4. sql查询——范围查询(区间查询)

    --范围查询(区间查询) --in() --查询年龄为18,28,38的人 select *from student where age=18 or age=28 or age=38; select ...

  5. adb logcat日志抓取

    adb命令 logcat日志抓取 一.logcat抓log方法:adb logcat命令,可以加条件过滤 1.安装SDK(参考android sdk环境安装) 2.使用数据线链接手机,在手机助手的sd ...

  6. 函数match应打印s中从ch1到ch2之间的所有字符,并且返回ch1的地址。

    1 char *match( char *s, char ch1, char ch2 ){ ; ; ; while(s[len]){ len++; } *len+];//防止s字符串全满导致t溢出 * ...

  7. Python_递归函数

    楔子 在讲今天的内容之前,我们先来讲一个故事,讲的什么呢?从前有座山,山里有座庙,庙里有个老和尚讲故事,讲的什么呢?从前有座山,山里有座庙,庙里有个老和尚讲故事,讲的什么呢?从前有座山,山里有座庙,庙 ...

  8. java_爬虫_获取经过js渲染后的网页源码

    md 弄了一天了……(这个月不会在摸爬虫了,浪费生命) 进入正题: 起初是想写一个爬虫来爬一个网站的视频,但是怎么爬取都爬取不到,分析了下源代码之后,发现源代码中并没有视频的dom 但是在浏览器检查元 ...

  9. docker dial unix /var/run/docker.sock: connect: permission denied

    Got permission denied while trying to connect to the Docker daemon socket at unix:///var/run/docker. ...

  10. Unity Coroutine详解(二)

    •        介绍•        Part 1. 同步等待•        Part 2. 异步协程•        Part 3. 同步协程•        Part 4. 并行协程 1.介绍 ...