java中的servlet采用的是单实例多线程方式
2014-02-14 17:46
295 查看
Servlet类本质上也是一个普通的类,并且Servlet容器默认只允许单个实例存在。当请求达到服务器的时候,Servlet实例如果已经存在的话则直接加载该实例,如果该Servlet类还未实例化则会先初始化这个Servlet。当请求到达Web服务器时,Web服务器中有一个线程池,它会从线程池中取一个工作线程,通过该线程调用请求的Servlet。因此,对Servlet来说,可以同时被好几个请求调用。请求结束后,线程放回线程池。
这种设计带来的好处是,Servlet单实例,减少了生成Servlet的开销。通过线程池响应请求,避免了不断创建线程和销毁线程的开销,提高了性能。但是这种多线程操纵单实例的模式,也会有一些副作用,那就是可能造成数据的不一致。看一个例子。
假设Servlet类如下:
public class HelloServlet extends HttpServlet
{
private String msg;
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException
{
msg = req.getParameter("msg");
try
{
Thread.sleep(10000);
}
catch(Exception ex)
{
ex.printStackTrace();
}
resp.getWriter().println(msg);
}
}
读取参数msg,然后等待10s后打印出来。
但是当2个用户在调用这个类的时候,就会有冲突了。
看下图的url,A用户先输入参数为hello,1秒钟之后,B用户后输入world。
显示的都是world。
原因很简单,因为两个线程都调用同一个实例,A用户把成员变量msg设成hello后,B用户又将他改成了world。
因此会有这种结果出现。
解决方法主要有以下3种,
1,使用Javax.servlet.SingleThreadModel(Servlet2.4中已经废弃该接口),此时Servlet容器将保证Servlet实例以单线程方式运行,也就是说,同一时刻,只会有一个线程执行Servlet的service()方法。
2.去除实例变量,使用局部变量。
3.使用同步代码块:synchronized{…}
最常用的方法还是第2种,因为他使用了局部变量,互不冲突,相比刻意的单线程和加锁机制,效率更高。因此代码只要如下修改即可。
public class HelloServlet extends HttpServlet
{
//private String msg;
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException
{
String msg = req.getParameter("msg");
try
{
Thread.sleep(10000);
}
catch(Exception ex)
{
ex.printStackTrace();
}
resp.getWriter().println(msg);
}
}
这种设计带来的好处是,Servlet单实例,减少了生成Servlet的开销。通过线程池响应请求,避免了不断创建线程和销毁线程的开销,提高了性能。但是这种多线程操纵单实例的模式,也会有一些副作用,那就是可能造成数据的不一致。看一个例子。
假设Servlet类如下:
public class HelloServlet extends HttpServlet
{
private String msg;
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException
{
msg = req.getParameter("msg");
try
{
Thread.sleep(10000);
}
catch(Exception ex)
{
ex.printStackTrace();
}
resp.getWriter().println(msg);
}
}
读取参数msg,然后等待10s后打印出来。
但是当2个用户在调用这个类的时候,就会有冲突了。
看下图的url,A用户先输入参数为hello,1秒钟之后,B用户后输入world。
显示的都是world。
原因很简单,因为两个线程都调用同一个实例,A用户把成员变量msg设成hello后,B用户又将他改成了world。
因此会有这种结果出现。
解决方法主要有以下3种,
1,使用Javax.servlet.SingleThreadModel(Servlet2.4中已经废弃该接口),此时Servlet容器将保证Servlet实例以单线程方式运行,也就是说,同一时刻,只会有一个线程执行Servlet的service()方法。
2.去除实例变量,使用局部变量。
3.使用同步代码块:synchronized{…}
最常用的方法还是第2种,因为他使用了局部变量,互不冲突,相比刻意的单线程和加锁机制,效率更高。因此代码只要如下修改即可。
public class HelloServlet extends HttpServlet
{
//private String msg;
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException
{
String msg = req.getParameter("msg");
try
{
Thread.sleep(10000);
}
catch(Exception ex)
{
ex.printStackTrace();
}
resp.getWriter().println(msg);
}
}
相关文章推荐
- java中的servlet采用的是单实例多线程方式
- Java ,单实例 多线程 ,web容器,servlet与struts1-2.x系列,线程安全的解决方案
- servlet采用单实例多线程模式开发的 (转)
- servlet的实例变量是线程不安全的,而其JSP也默认是以多线程方式执行(原创)
- Java项目 1 - 学生成绩管理系统---采用 Servlet+Jsp+JavaBean+MySql 设计方式,
- Java--servlet + ThreadLocal解决多线程并发问题及实例
- servlet的实例变量是线程不安全的,而其JSP也默认是以多线程方式执行
- Java ,单实例 多线程 ,web容器,servlet与struts1-2.x系列,线程安全的解决
- servlet2.x采用的是单实例多线程方式
- servlet响应多个请求——servlet采用单实例多线程模式开发的
- java线程基础巩固---采用多线程方式模拟银行排队叫号以及Runnable接口存在的必要性
- servlet采用单实例多线程模式怎样确保线程安全
- java多线程的三种实现方式
- 实现java多线程的3种方式,99%人没用过第3种
- JAVA多线程实现的三种方式
- Java多线程实例->生产消费模式
- (一) Java多线程详解之创建线程的两种方式
- JAVA多线程实现的三种方式
- java 多线程实例二
- Java多线程-实例解析