最近在看线程的有关知识,碰到一个小问题,目前还没有解决,现记录下来。



如果在我们自己写的servlet里有成员变量,因为多线程的访问就会出现一些线程问题。这点大家都知道,我们看下面的例子。

public class ConcurrentTest extends HttpServlet {
    PrintWriter output;
    @Override
    protected void service(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        String  username;
        response.setContentType("text/html;charset=gb2312");
        username=request.getParameter("Username");
        output=response.getWriter();
        try {
            //为了突出并发问题,在这设置一个延时
            Thread.sleep(5000);
            output.println("用户名:"+username+"<BR>");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

部署在tomcat7上

然后我们在浏览器上发两个链接。

  a: http://localhost:8080/ServletTest/ConcurrentTest?Username=a

  b: http://localhost:8080/ServletTest/ConcurrentTest?Username=b

当a发送后,过半秒后发送b。

我们可以看到a请求的返回页是什么都没有

而b上显示

用户名:a

用户名:b



为什么会有这个效果是因为,a请求的output这个变量在多线程的情况下被b线程的output覆盖了。所以"两个"(其实只有一个)servlet的输出流对象都定位到b上面了。



这是一个经典的例子。

再看下面这个。

public class SimpleServlet extends HttpServlet{
       // A variable that is NOT thread-safe!
       private int counter = 0;
       public void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
              doPost(req, resp);
       }

       public void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException
       {
              resp.getWriter().println("<HTML><BODY>");
              resp.getWriter().println(this + " ==> ");
              resp.getWriter().println(Thread.currentThread() + ": <br>");
              for (int c = 0; c < 10; c++){
                     resp.getWriter().println("Counter = " + counter + "<BR>");
                     try {
                            Thread.sleep((long) Math.random() * 1000);
                            counter++;
                     }
                     catch (InterruptedException exc){}
              }
              resp.getWriter().println("</BODY></HTML>");
       }
}

和第一个例子似乎差不多,只是变量由PrintWriter变成了int。

我们在做三个请求。

    <HTML>  

    <BODY>  

    <TABLE>
        <TR>
            <TD><IFRAME src="./SimpleServlet" name="servlet1" height="200%"> </IFRAME></TD>
        </TR>
        <TR>
            <TD><IFRAME src="./SimpleServlet" name="servlet2" height="200%"> </IFRAME></TD>
        </TR>
        <TR>
            <TD><IFRAME src="./SimpleServlet" name="servlet3" height="200%"> </IFRAME></TD>
        </TR>
    </TABLE>
    </BODY>
    </HTML>  

在tomcat7下 结果是

com.zwchen.servlet.SimpleServlet@109da93:

Counter = 0

Counter = 1

Counter = 2

Counter = 3

Counter = 4

Counter = 5

Counter = 6

Counter = 7

Counter = 8

Counter = 9





com.zwchen.servlet.SimpleServlet@109da93:

Counter = 10

Counter = 11

Counter = 12

Counter = 13

Counter = 14

Counter = 15

Counter = 16

Counter = 17

Counter = 18

Counter = 19



com.zwchen.servlet.SimpleServlet@109da93:

Counter = 20

Counter = 21

Counter = 22

Counter = 23

Counter = 24

Counter = 25

Counter = 26

Counter = 27

Counter = 28

Counter = 29

并没有出现我们想象中的乱序问题。



一种解释是tomcat7中使用了ThreadPoolExecutor技术。

但是我依然不明白,那为什么同是在tomcat7下,第一个例子就会出现覆写的问题呢?

这个问题,估计的看源码才行。

如果有哪位大侠知道问题的答案请告诉我,跪谢了。

感谢glt

参考资料

两个例子分别来自下面两个地址 详细信息大家自己去看看吧

http://www.cnblogs.com/gw811/archive/2012/09/07/2674859.html

http://zwchen.iteye.com/blog/91088

tomcat中的线程问题2的更多相关文章

  1. tomcat中的线程问题

    看这篇文章之前,请先阅读: how tomcat works 读书笔记 十一 StandWrapper 上 地址如下: http://blog.csdn.net/dlf123321/article/d ...

  2. Tomcat中常见线程说明

    http://blog.csdn.NET/jeff_fangji/article/details/41786205 本文讲述了Tomcat的常见线程的功能.名称.线程池和配置等信息,其中源码来自于To ...

  3. Tomcat中的线程池StandardThreadExecutor

    之所以今天讨论它,因为在motan的的NettyServer中利用它这个线程池可以作为业务线程池,它定制了一个自己的线程池.当然还是基于jdk中的ThreadExecutor中的构造方法和execut ...

  4. 详解tomcat连接数和线程数

    前言 在使用tomcat时,经常会遇到连接数.线程数之类的配置问题,要真正理解这些概念,必须先了解Tomcat的连接器(Connector). 在前面的文章 详解Tomcat配置文件server.xm ...

  5. Tomcat 连接数与线程池详解

    前言 在使用tomcat时,经常会遇到连接数.线程数之类的配置问题,要真正理解这些概念,必须先了解Tomcat的连接器(Connector). 在前面的文章 详解Tomcat配置文件server.xm ...

  6. Tomcat中JVM内存溢出及合理配置及maxThreads如何配置(转)

    来源:http://www.tot.name/html/20150530/20150530102930.htm Tomcat本身不能直接在计算机上运行,需要依赖于硬件基础之上的操作系统和一个Java虚 ...

  7. e.Tomcat中的sendfile支持

    sendfile实质是linux系统中一项优化技术,用以发送文件和网络通信时,减少用户态空间与磁盘倒换数据,而直接在内核级做数据拷贝,这项技术是linux2.4之后就有的,现在已经很普遍的用在了C的网 ...

  8. tomcat中server.xml配置详解

    Tomcat Server的结构图如下: 该文件描述了如何启动Tomcat Server <Server>     <Listener />     <GlobaNami ...

  9. How Tomcat works — 八、tomcat中的session管理

    在使用shiro的session的时候感觉对于tomcat中session的管理还不是特别清楚,而且session管理作为tomcat中比较重要的一部分还是很有必要学习的. 目录 概述 session ...

随机推荐

  1. Redis 学习笔记3:Jedis 连接虚拟机下的Redis 服务

    Jedis 是 Redis 官方首选的 Java 客户端开发包. 虚拟机的IP地址是192.168.8.88. Jedis代码是放在windows上的,启动虚拟机上的Redis服务之后,用Jedis连 ...

  2. 找不到BufferedImage这个Class的解决方法

    找不到BufferedImage这个Class的解决方法 环境: [1]RedHat AS5 64位      [2]WebSphere6.0 32位版本 正文:    发现原来在RedHat AS4 ...

  3. maven跳过单元测试的两个参数区别

    maven在打包过程中需要执行单元测试.但有些时候单元测试已经通过只是想打包时,想跳过测试.maven提供了两个参数跳过测试:maven.test.skip=true 和skipTests. 例子 m ...

  4. 剑指Offer——京东校招笔试题+知识点总结

    剑指Offer--京东校招笔试题+知识点总结 笔试感言 经过一系列的笔试,发觉自己的基础知识还是比较薄弱的,尤其是数据结构和网络,还有操作系统.工作量还是很大的.做到精确制导的好方法就是在网上刷题,包 ...

  5. 02_c3p0之c3p0-config.xml配置案例,操作c3p0的jdbcUtil工具类的编写

     c3p0也是一个开源jdbc连接池,我们熟悉的Hibernate和Spring框架使用的都是该数据源. 这里获得数据源使用的方法是:ComboPooledDataSource 它提供的构造方法有 ...

  6. Retrofit 2.0 超能实践(一),okHttp完美支持Https传输

    http: //blog.csdn.net/sk719887916/article/details/51597816 Tamic首发 前阵子看到圈子里Retrofit 2.0,RxJava(Andro ...

  7. Android中Socket通信之TCP与UDP传输原理

    一.Socket通信简介 Android与服务器的通信方式主要有两种,一是Http通信,一是Socket通信.两者的最大差异在于,http连接使用的是"请求-响应方式",即在请求时 ...

  8. 让 Google Test 出错时断点

    Google Test 缺省是出错退出. 如果最后的出错行在系统库中,那就没什么帮助. 如果是调试运行,直接退出根本就不知道哪里出错了. 后来添加了一个运行参数: --gtest_break_on_f ...

  9. J2EE进阶(十一)SSH框架整合常见问题汇总(二)

    org.hibernate.PropertyAccessException: IllegalArgumentException occurred while calling setter of cn. ...

  10. SDL2源代码分析5:更新纹理(SDL_UpdateTexture())

    ===================================================== SDL源代码分析系列文章列表: SDL2源代码分析1:初始化(SDL_Init()) SDL ...