对于多线程下Servlet以及Session的一些理解
今天,小伙伴突然问到了Servlet是不是线程安全的问题。脑子当时一卡壳,只想到了单实例多线程。这里做一些总结。
Servlet体系是建立在Java多线程的基础之上的,它的生命周期是由Tomcat来维护的。当客户端第一次请求Servlet的时候,tomcat会根据web.xml配置文件实例化servlet,当又有一个客户端访问该servlet的时候,不会再实例化该servlet,也就是多个线程在使用这个实例。
Servlet线程池
serlvet采用多线程来处理多个请求同时访问,Tomcat容器维护了一个线程池来服务请求。
线程池实际上是等待执行代码的一组线程叫做工作组线程(Worker Thread),Tomcat容器使用一个
调度线程来管理工作组线程(Dispatcher Thead)。
当容器收到一个Servlet请求,Dispatcher线程从线程池中选出一个工作组线程,将请求传递
给该线程,然后由该线程来执行Servlet的service方法。
当这个线程正在执行的时候,容器收到另一个请求,调度者线程将从线程池中选出另外一个
工作组线程来服务则个新的请求,容器并不关心这个请求是否访问的是同一个Servlet还是另一个
Servlet。当容器收到对同一个Servlet的多个请求的时候,那这个servlet的service方法将在多线程
中并发的执行。
Servlet线程安全问题
多线程和单线程Servlet具体区别:多线程下每个线程对局部变量都会有自己的一份copy,这
样对局部变量的修改只会影响到自己的copy而不会对别的线程产生影响,线程安全的。但是对于
实例变量来说,由于servlet在Tomcat中是以单例模式存在的,所有的线程共享实例变量。多个线程
对共享资源的访问就造成了线程不安全问题。
那么如何使Servlet线程安全呢?
设计线程安全的Servlet
针对上述的情况如何设计线程安全的Servlet呢?我们知道的是多线程是不共享局部变量的
servlet线程不安全也是针对于共享资源的访问才产生的。 因此这里就有一种方式了。
变量的线程安全
这里的变量变量指的是字段和共享数据,主要是表单的参数值。基于多线程不共享局部变量的
特点我们可以将这类变量参数本地化。例如对于上面的一个实例我们可以这样设计
protected void doPost(HttpServletRequest request,
HttpServletResponse response) throws ServletException, IOException {
String message;
message = request.getParameter("message");
PrintWriter printWriter = response.getWriter();
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
printWriter.write(message);
}
属性的线程安全
ServletContext:它是线程不安全的,多线程下可以同时进行读写,因此我们要对其读写操作进行
同步或者深度的clone。
HttpSession:同样是线程不安全的,和ServletContext的操作一样。
ServletRequest:它是线程安全的,对于每一个请求由一个工作线程来执行,都会创建一个
ServletRequest对象,所以ServletResquest只能在一个线程中被访问,而且他只在service()方法内是
有效的。
同步的集合类
在使用java中的集合API进行处理的时候,选择同步的集合。
外部对象互斥
在多个Servlet中对某个外部对象(例如文件)的修改是务必加锁,互斥访问。不过这里需要注意的是
使用Synchronized的时候这意味着线程需要排队等待处理,因此在使用同步块的时候要尽量的缩小同
步块的代码范围。不要直接在方法上用同步,这样会严重影响性能。
值得一提的是最好别再serlvet中创建自己的线程来完成某个功能,这会是情况更加复杂。
Single ThreadMode接口
这也是解决servlet线程安全问题的一个方法,Single ThreadMode是一个标识接口,如果一个Servlet
实现了该接口,那么Tomcat将保证在一个时刻仅有一个线程可以在给定的Serlvet实例的service方法中
执行。其他所有请求进行排队。(针对单个实例)
可以看出的是这种方式虽然可以解决线程安全问题,可以效率太过低下。
其再Servlet的规范中已经被废弃了。
总结
Servlet的线程安全问题只有在大量的并发访问时才会显现出来,并且很难发现,因此在编写Servlet程序
时要特别注意。线程安全问题主要是由实例变量造成的,因此在Servlet中应避免使用实例变量。如果应用程
序设计无法避免使用实例变量,那么使用同步来保护要使用的实例变量,但为保证系统的最佳性能,应该
同步可用性最小的代码路径。
下面来说第二部分: HttpSession
http://blog.csdn.net/haitao111313/article/details/7735620
对于多线程下Servlet以及Session的一些理解的更多相关文章
- Java并发之多线程下竞态条件概念的理解
一.简述 竞态条件(Race Condition):计算的正确性取决于多个线程的交替执行时序时,就会发生竞态条件. 二.常见竞态条件分析 最常见的竞态条件为 1.先检测后执行 执行依赖于检测的结果,而 ...
- 关于Spring在多线程下的个人疑问
在Web开发中,不可避免的是需要遇到并发操作的,并发操作就有可能会引发我们的多线程安全问题.比如说,我们多线程下访问同一个变量并且有一个线程做出修改那么就会使得我们另外的线程在不知情的情况下被修改自己 ...
- Java进阶专题(十五) 从电商系统角度研究多线程(下)
前言 本章节继上章节继续梳理:线程相关的基础理论和工具.多线程程序下的性能调优和电商场景下多线程的使用. 多线程J·U·C ThreadLocal 概念 ThreadLocal类并不是用来解决 ...
- 多线程下NSOperation、NSBlockOperation、NSInvocationOperation、NSOperationQueue的使用
本篇文章主要介绍下多线程下NSOperation.NSBlockOperation.NSInvocationOperation.NSOperationQueue的使用,列举几个简单的例子. 默认情况下 ...
- servlet的session为null?
servlet的session(会话)显示为null,一般是web.xml中配置不对或者在浏览器输入的url不正确造成的. web.xml配置如下: <servlet> <servl ...
- python 类变量 在多线程下的共享与释放问题
最近被多线程给坑了下,没意识到类变量在多线程下是共享的,还有一个就是没意识到 内存释放问题,导致越累越大 1.python 类变量 在多线程情况 下的 是共享的 2.python 类变量 在多线程情况 ...
- Java多线程21:多线程下的其他组件之CyclicBarrier、Callable、Future和FutureTask
CyclicBarrier 接着讲多线程下的其他组件,第一个要讲的就是CyclicBarrier.CyclicBarrier从字面理解是指循环屏障,它可以协同多个线程,让多个线程在这个屏障前等待,直到 ...
- Java多线程20:多线程下的其他组件之CountDownLatch、Semaphore、Exchanger
前言 在多线程环境下,JDK给开发者提供了许多的组件供用户使用(主要在java.util.concurrent下),使得用户不需要再去关心在具体场景下要如何写出同时兼顾线程安全性与高效率的代码.之前讲 ...
- 多线程下C#如何保证线程安全?
多线程编程相对于单线程会出现一个特有的问题,就是线程安全的问题.所谓的线程安全,就是如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码.如果每次运行结果和单线程运行的结果是 ...
随机推荐
- Windows Azure Virtual Network (11) 虚拟网络之间点对点连接VNet Peering
<Windows Azure Platform 系列文章目录> 在有些时候,我们需要通过VNet Peering,把两个虚拟网络通过内网互通互联.比如: 1.在订阅A里的Virtual N ...
- 小甲鱼OD学习第11讲
这次我们的任务是破解这个需要注册的软件,如下图所示 我们这次从字符串入手,我们查找 unregistered 字符串 然后我们在如下图的字符串下断点 然后我们来到断点处,我们观察到 地址为 0040 ...
- python进行各类API的使用
前言: 献上歌曲一首: 因为快要上学了,昨天晚上熬夜.然后今天早上起床 没有什么精神.吃完午饭后开始思考今天写什么好呢 然后逛着逛着逛到了一个API网站.感觉还不错就爬了 0x01: 环境:windo ...
- ThinkPHP的基本操作
一.生成入口文件 1.打开服务器,在本地环境测试时在地址栏输入localhost/项目文件名/index.php 可以在Application下面生成一个home模块,记得在这之前,要建立一个项目文 ...
- GitHub入门与实践
基本命令 git status 工作区状态 git add git commint 暂存区 git push gitHub客户端 下载网址:https://desktop.github.com/ 解决 ...
- CentOS源码安装Python3.6
一.安装环境及版本 CentOS 6.5 Python 3.6.1 二.安装依赖包 1.安装静态库 # yum install -y openssl-static 注:如果不安装该静态库,会导致pyt ...
- [记]WIndow/Linux 获取本机(全部)IPv4、IPv6、MAC地址方法 (C/C++)
Linux 获取本机IP.MAC地址用法大全 //#include <sys/types.h> #include <ifaddrs.h> #include <sys/io ...
- 在gitlab上面创建私有库
一.创建私有库1.使用xcode建立新的工程,选择Cocoa Touch Static Library,取名为podTest WechatIMG1172.jpeg 2.创建一个类PodTest,给 ...
- TP5整合 WorkerMan 以及 GatewayWorker
TP5整合GatewayWorker Windows版安装 a)使用composer create-project topthink/think testTG,来安装thinkphp5. b)进入t ...
- ios设备触发虚拟键盘输入后position:fixed 无效的一些简单另类的解决方法。
首先看一下我要解决的问题,第一张图是正常的情况下,第二张图是点击了输入框之后的情况,就是要解决此问题~! 百度了一下解决方法,好像有以下的一些方法: 1. iscroll 2. Jquery Mobi ...