Servlet及JSP中的多线程同步问题

js|servlet|多线程|问题

  Servlet/JSP技术和ASP、PHP等相比,由于其多线程运行而具有很高的执行效率。由于Servlet/JSP默认是以多线程模式执行的,所以,在编写代码时需要非常细致地考虑多线程的同步问题。然而,很多人编写Servlet/JSP程序时并没有注意到多线程同步的问题,这往往造成编写的程序在少量用户访问时没有任何问题,而在并发用户上升到一定值时,就会经常出现一些莫明其妙的问题,对于这类随机性的问题调试难度也很大。

  一、在Servlet/JSP中的几种变量类型

  在编写Servlet/JSP程序时,对实例变量一定要小心使用。因为实例变量是非线程安全的。在Servlet/JSP中,变量可以归为下面的几类:

  1. 类变量

  request,response,session,config,application,以及JSP页面内置的page, pageContext。其中除了application外,其它都是线程安全的。

  2. 实例变量

  实例变量是实例所有的,在堆中分配。在Servlet/JSP容器中,一般仅实例化一个Servlet/JSP实例,启动多个该实例的线程来处理请求。而实例变量是该实例所有的线程所共享,所以,实例变量不是线程安全的。

  3. 局部变量

  局部变量在堆栈中分配,因为每一个线程有自己的执行堆栈,所以,局部变量是线程安全的。

  二、在Servlet/JSP中的多线程同步问题

  在JSP中,使用实例变量要特别谨慎。首先请看下面的代码:

ffffff cellPadding=2 width=540 align=center borderColorLight=black border=1>

// instanceconcurrenttest.jsp<%@ page contentType="text/html;charset=GBK" %><%!     //定义实例变量     String username;     String password;    java.io.PrintWriter output;%><%     //从request中获取参数    username = request.getParameter("username");    password = request.getParameter("password");    output = response.getWriter();    showUserInfo();  %> <%!     public void showUserInfo() {         //为了突出并发问题,在这儿首先执行一个费时操作         int i =0;         double sum = 0.0;         while (i++ < 200000000) {             sum += i;         }                  output.println(Thread.currentThread().getName() + "<br>");       output.println("username:" + username + "<br>");        output.println("password:" + password + "<br>");     } %>

  在这个页面中,首先定义了两个实例变量,username和password。然后在从request中获取这两个参数,并调用showUserInfo()方法将请求用户的信息回显在该客户的浏览器上。在一个用户访问是,不存在问题。但在多个用户并发访问时,就会出现其它用户的信息显示在另外一些用户的浏览器上的问题。这是一个严重的问题。为了突出并发问题,便于测试、观察,我们在回显用户信息时执行了一个模拟的费时操作,比如,下面的两个用户同时访问(可以启动两个IE浏览器,或者在两台机器上同时访问):

a: http://localhost:8080/instanceconcurrenttest.jsp?username=a&password=123

b: http://localhost:8080/instanceconcurrenttest.jsp?username=b&password=456

如果a点击链接后,b再点击链接,那么,a将返回一个空白屏幕,b则得到a以及b两个线程的输出。请看下面的屏幕截图:

图1:a的屏幕

图2:b的屏幕

  从运行结果的截图上可以看到,Web服务器启动了两个线程分别来处理来自a和b的请求,但是在a却得到一个空白的屏幕。这是因为上面程序中的output, username和password都是实例变量,是所有线程共享的。在a访问该页面后,将output设置为a的输出,username,password分别置为a的信息,而在a执行printUserInfo()输出username和password信息前,b又访问了该页面,把username和password置为了b的信息,并把输出output指向到了b。随后a的线程打印时,就打印到了b的屏幕了,并且,a的用户名和密码也被b的取代。请参加下图所示:

图3:a、b两个线程的时间线

  而实际程序中,由于设置实例变量,使用实例变量这两个时间点非常接近,所以,像本例的同步问题并没有这么突出,可能会偶尔出现,但这却更加具有危险性,也更加难于调试。

  同样,对于Servlet也存在实例变量的多线程问题,请看上面页面的Servlet版:

// InstanceConcurrentTest.javaimport javax.servlet.*;import javax.servlet.http.*;import java.io.PrintWriter;public class InstanceConcurrentTest extends HttpServlet {       String username;       String password;       PrintWriter out;       public void doGet(HttpServletRequest request,        HttpServletResponse response)                 throws ServletException,java.io.IOException        {             //从request中获取参数      username = request.getParameter("username");    password = request.getParameter("password");     System.out.println(Thread.currentThread().getName() +     " | set username:" + username);    out = response.getWriter();     showUserInfo();         }        public void showUserInfo() {             //为了突出并发问题,在这儿首先执行一个费时操作      int i =0;      double sum = 0.0;      while (i++ < 200000000) {       sum += i;      }      out.println("thread:" + Thread.currentThread().getName());    out.println("username:"+ username);     out.println("password:" + password);  }}

  三、解决方案

  1. 以单线程运行Servlet/JSP

  在JSP中,通过设置:<%@ page isThreadSafe="false" %>,在Servlet中,通过实现javax.servlet.SingleThreadModel,此时Web容器将保证JSP或Servlet实例以单线程方式运行。

  重要提示:在测试中发现,Tomcat 4.1.17不能正确支持isThreadSafe属性,所以,指定isTheadSafe为false后,在Tomcat 4.1.17中仍然出现多线程问题,这是Tomcat 4.1.17的Bug。在Tomcat 3.3.1和Resin 2.1.5中测试通过。

  2. 去除实例变量,通过参数传递

  从上面的分析可见,应该在Servlet/JSP中尽量避免使用实例变量。比如,下面的修正代码,去除了实例变量,通过定义局部变量,并参数进行传递。这样,由于局部变量是在线程的堆栈中进行分配的,所以是线程安全的。不会出现多线程同步的问题。代码如下:

<%@ page contentType="text/html;charset=GBK" %><%     //使用局部变量    String username;    String password;    java.io.PrintWriter output;    //从request中获取参数    username = request.getParameter("username");    password = request.getParameter("password");    output = response.getWriter();    showUserInfo(output, username, password);%><%!     public void showUserInfo(java.io.PrintWriter _output,     String _username, String _password) {    //为了突出并发问题,在这儿首先执行一个费时操作      int i =0;       double sum = 0.0;       while (i++ < 200000000) {         sum += i;        }          _output.println(Thread.currentThread().getName() + "<br>");      _output.println("username:" + _username + "<br>");      _output.println("password:" + _password + "<br>");      }%>

注:有的资料上指出在printUserInfo()方法或者实例变量的相关操作语句上使用synchronized关键字进行同步,但这样并不能解决多线程的问题。因为,这样虽然可以使对实例变量的操作代码进行同步,但并不能阻止一个线程使用另外一个线程修改后的“脏的”实例变量。所以,除了降低运行效率外,不会起到预期效果。

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索jsp问题
, servlet
, 问题
, 线程
, 实例
, 多线程
, 变量
, 多线程问题
, 两个变量
, 多线程实例
同步变量
servlet多线程问题、多线程同步问题、servlet 多线程、servlet是多线程的吗、servlet单实例多线程,以便于您获取更多的相关知识。

时间: 2024-10-23 21:24:05

Servlet及JSP中的多线程同步问题的相关文章

关于Servlet、Jsp中的多国语言显示 -- 转自高人

js|servlet|显示 /***有朋友有如此苦心之作,实在感谢,相信它不会怪我转贴吧,他的email是vividq@china.com*/ 关于Servlet.Jsp中的多国语言显示 因为一直不信Java竟会有不能混排显示多国语言的BUG,这个周末研究了一下Servlet.Jsp的多国语言显示的问题,也就是Servlet的多字符集问题,由于我对字符集的概念还不是很清晰所以写出的东西未必是准确的,我是这样理解Java中的字符集的:在运行时,每个字符串对象中存储的都是编码为UNICODE内码的(

关于Servlet、Jsp中的多国语言显示

js|servlet|显示 因为一直不信Java竟会有不能混排显示多国语言的BUG,这个周末研究了一下Servlet.Jsp的多国语言显示的问题,也就是Servlet的多字符集问题,由于我对字符集的概念还不是很清晰所以写出的东西未必是准确的,我是这样理解Java中的字符集的:在运行时,每个字符串对象中存储的都是编码为UNICODE内码的(我觉得所有的语言中都是有相应编码的,因为在计算机内部字符串总是用内码来表示的,只不过一般计算机语言中的字符串编码时平台相关的,而Java则采用了平台无关的UNI

在Servlet与JSP中取得当前文件所在的相对路径与绝对路径

js|servlet //Servlet中 //JSP中的application对象就是Servlet中的ServerContext,所以在Servlet中是如此获得 //import java.io.File; System.out.println("根目录所对应的绝对路径:" + request.getServletPath() + ""); String strPathFile = request.getSession().getServletContext(

关于Servlet、Jsp中的多国语言显示(续)

js|servlet|显示 续   --- /*** Insert the method's description here.* Creation date: (2001-2-4 17:28:17)* @return java.util.Enumeration*/public Enumeration getParameterNames() {if (pairs == null) return null;return pairs.keys();}/*** Insert the method's

Servlet、Jsp中的多国语言显示

js|servlet|显示 因为一直不信Java竟会有不能混排显示多国语言的BUG,这个周末研究了一下Servlet.Jsp的多国语言显示的问题,也就是Servlet的多字符集问题,由于我对字符集的概念还不是很清晰所以写出的东西未必是准确的,我是这样理解Java中的字符集的:在运行时,每个字符串对象中存储的都是编码为UNICODE内码的(我觉得所有的语言中都是有相应编码的,因为在计算机内部字符串总是用内码来表示的,只不过一般计算机语言中的字符串编码时平台相关的,而Java则采用了平台无关的UNI

浅析Servlet和Jsp中的多国语言显示

因为一直不信Java竟会有不能混排显示多国语言的BUG,这个周末研究了一下Servlet.Jsp的多国语言 显示的问题,也就是Servlet的多字符集问题,由于我对字符集的概念还不是很清晰所以写出的东西未必 是准确的,我是这样理解Java中的字符集的:在运行时,每个字符串对象中存储的都是编码为UNICODE内 码的(笔者认为所有的语言中都是有相应编码的,因为在计算机内部字符串总是用内码来表示的,只不过 一般计算机语言中的字符串编码时平台相关的,而Java则采用了平台无关的UNICODE). Ja

jsp中获取绝对路径和相对路径方法

//在Servlet与JSP中取得当前文件所在的相对路径与绝对路径   JSP中 根目录所对应的绝对路径:request.getRequestURI() 文件的绝对路径 :application.getRealPath(request.getRequestURI()); 当前web应用的绝对路径 :application.getRealPath("/"); 取得请求文件的上层目录:new File(application.getRealPath(request.getRequestUR

JSP 中的servlet在本地能运行,在服务器端不能运行

问题描述 JSP 中的servlet在本地能运行,在服务器端不能运行 做一个用户名密码的登陆表单,使用servlet对输入的用户名密码与数据库里用户名密码进行比较,如果正确,就转至登陆页面,不正确就提示错误.在本地运行一切正常,但是放在服务器上,输入完用户名密码.点击确定后,就一直是网页正在连接的状态,死活不能运行出来.求大神看看到底是哪里有问题.(对不起实在没有C币了--) 解决方案 惊喜!Java为服务器端Web应用带来最高运行速度 解决方案二: 自己看一下日志文件,有没有报错信息. 据你描

从jsp中向servlet传数据库指针的写法不知道对不对,rs是不是可以传过去,求大神指导

问题描述 从jsp中向servlet传数据库指针的写法不知道对不对,rs是不是可以传过去,求大神指导 rs是数据库指针,想在servlet中获取rs数据库指针,不知道这么写对不对