锁定老帖子 主题:Servlet的线程安全问题
精华帖 (2) :: 良好帖 (0) :: 新手帖 (7) :: 隐藏帖 (0)
|
|
---|---|
作者 | 正文 |
发表时间:2008-08-08
最后修改:2008-11-29
这个问题网上一直没有搜到很详细的解释,也可能是高人的解释不符合我的理解方式。所以自己到网上搜集了写资料再加自己的想法,随便写了点东西发到论坛上,希望大家给予修正意见,看我是否理解对了。 new Runnalbe{ public run(){ Request requset = new Request(); Resposne response = new Response(); //servlet对象只有一个,是容器自动生成的,这里模拟一个servlet的调用过程。 servlet.doPost (request,response); } }
public class HelloServlet extends HttpServlet { private int j =0; public void doPost(HttpServletRequest request, HttpServletResponse response){ int i=0; i++; j++; //这里的i和j那个是线程安全的那个不是呢,后面我们将从线程的堆栈,和jvm的堆的概念来解释这个问题 //request 和 response 对象是不是线程安全的 } }
现在我们回到开始那个例子,可以很好的分析出参数request,respone,i是线程安全的,而j是线程不安全的; 为什么?request,respone是线程安全的是因为每个线程对应的request,respone对象都是不一样的,不存在共享问题。i是线程安全的是应为i是局部变量,每个线程的堆栈中存放的值也是各自独立的。 j线程不安全是应为它是类HelloServlet的属性,找了很多资料都不能说清楚它到底放在那里,从实际效果来看,它是不安全的,所以应该是放在和类对象一起放在堆里面的,堆里面估计是复制了一份过来,因为HelloServlet对多个线程而言只有 一个实例,所以存在共享问题。
声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
发表时间:2008-08-11
其实说简单一点就是:
servlet是单实例多线程运行方式,所以对象变量线程不安全,局部变量线程安全 |
|
返回顶楼 | |
发表时间:2008-08-13
引用 JVM为每个新创建的线程都分配一个堆栈.
这个应该是错的,栈也许分配了(这个有待高人解释清楚),毕竟有一个调用栈。但是堆如果重新分配一个,一则消耗太大,二来根本就不会出现数据共享的问题了,各线程都独立的堆了,还会有线程安全问题? 局部变量不出现线程安全问题不是因为在各自堆里面,而是不同线程的局部变量都是调用时才生成的,压根就不是同一个对象,不同线程的引用分别指向了堆中不同的对象,当然不会问题;对象的变量因为servlet是单例,不同线程的引用全部指向同一个对地址,才会有共享数据问题。 lz所说的复制堆数据,那基本只有在gc时,从young区往old区挪的时候才会干 |
|
返回顶楼 | |
发表时间:2008-08-18
nowonder 写道 引用 JVM为每个新创建的线程都分配一个堆栈.
这个应该是错的,栈也许分配了(这个有待高人解释清楚),毕竟有一个调用栈。但是堆如果重新分配一个,一则消耗太大,二来根本就不会出现数据共享的问题了,各线程都独立的堆了,还会有线程安全问题? 局部变量不出现线程安全问题不是因为在各自堆里面,而是不同线程的局部变量都是调用时才生成的,压根就不是同一个对象,不同线程的引用分别指向了堆中不同的对象,当然不会问题;对象的变量因为servlet是单例,不同线程的引用全部指向同一个对地址,才会有共享数据问题。 lz所说的复制堆数据,那基本只有在gc时,从young区往old区挪的时候才会干 我这里说的堆栈就是你说的栈,而不是一般理解的堆。这里写的不准确,不好意思。 局部变量应该是存放在线程的堆栈(栈)里面,局部变量总有个存放的地方吧,不放栈里难道放堆里?当然栈里面放的只是对象的应用啦,对象还是放在堆中的。 复制数据这个问题,我理解的也不是太清楚,但是对于变量j而言,如果不复制一份到栈里面,而是多个线程共享的话。在j++是,其实相当于j=j+1 这个j临时变量存放在哪里呢?应该是存放在栈中,既然j存放在栈中,而原始的j数据存放在堆中,当然存在数据复制问题。 |
|
返回顶楼 | |
发表时间:2008-08-28
一般我们新建对象,jvm 才会为我们新建存放对象的堆空间,里面有属于对象的方法体和属性,而对对象的引用和基础数据类型是存放在栈中的,而cpu运作时,进行运算而产生的中间结果会放在cpu的寄存器中。
所以基础数据声明后,不管怎么赋值,都不会有线程问题(int i=1;int j=i)两个int 类型的数据根本就是2个数据体。 局部变量本来就是在某个线程的生命周期内声明的变量,如果不用特别的方法保存它,其他线程根本没办法引用到它,所以也就没有线程同步的问题存在 而引用同一个对象,修改得是同一个堆中的数据,线程同步腐蚀就出现了,如果没有状态数据,被使用对象也就没有被线程修改得可能,也就没有线程安全问题 |
|
返回顶楼 | |
发表时间:2008-08-28
地址引用,也可以归类到堆栈这些东西里面把
|
|
返回顶楼 | |
发表时间:2008-08-28
你例子举的不好,概念也不完全对
其实就是一个线程安全的问题 和servlet没有关系 你这个j是否线程安全取决于你的servlet是否是同一个对象 而servlet是否是单例取决于容器的实现 一般来说不是单例的 |
|
返回顶楼 | |
发表时间:2008-08-29
allenofchina 写道 而servlet是否是单例取决于容器的实现 一般来说不是单例的 不对吧,一般来说Servlet都是单例的。 public class TestServletSingle extends HttpServlet { protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } response.getOutputStream().println(this.hashCode()); } protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } } 你在5秒内打开N个浏览器窗口试试,返回的都是相同的Hashcode。只有在context reload以后,Hashcode才发生变化。因为原来的servlet object销毁了,容器又新建了一个。 然而,Servlet的单例不由class来决定,而是由您在web.xml里配置的servlet-name来决定。也就是说每一个servlet-name只有1个单例。当你有3个不同的serlvet-name指定了相同的1个servlet-class时,容器会产生3个不同的servlet object。只有3个。 |
|
返回顶楼 | |
发表时间:2008-08-29
楼上写得不错
我也没有验证过 |
|
返回顶楼 | |
发表时间:2009-01-01
最后修改:2009-01-01
1)所以多线程并发的情况下,出现不同步的现象主要是因为各自堆栈存放的某些数据是共享的,说白了就是同一个数据的引用(不是copy)被存放在不同的堆栈中。
准确点说是堆数据共享 2)例如类X的对象A被多个线程访问,他的引用被保存在多个线程的堆栈中, 一个java虚拟机中只有一个堆被多个线程共享 |
|
返回顶楼 | |