Servlet/JSP技術(shù)和ASP、PHP等相比,由于其多線(xiàn)程運行而具有很高的執行效率。由于Servlet/JSP默認是以多線(xiàn)程模式執行的,所以,在編寫(xiě)代碼時(shí)需要非常細致地考慮多線(xiàn)程的同步問(wèn)題。然而,很多人編寫(xiě)Servlet/JSP程序時(shí)并沒(méi)有注意到多線(xiàn)程同步的問(wèn)題,這往往造成編寫(xiě)的程序在少量用戶(hù)訪(fǎng)問(wèn)時(shí)沒(méi)有任何問(wèn)題,而在并發(fā)用戶(hù)上升到一定值時(shí),就會(huì )經(jīng)常出現一些莫明其妙的問(wèn)題,對于這類(lèi)隨機性的問(wèn)題調試難度也很大。
一、在Servlet/JSP中的幾種變量類(lèi)型
在編寫(xiě)Servlet/JSP程序時(shí),對實(shí)例變量一定要小心使用。因為實(shí)例變量是非線(xiàn)程安全的。在Servlet/JSP中,變量可以歸為下面的幾類(lèi):
1. 類(lèi)變量
request,response,session,config,application,以及JSP頁(yè)面內置的page, pageContext。其中除了application外,其它都是線(xiàn)程安全的。
2. 實(shí)例變量
實(shí)例變量是實(shí)例所有的,在堆中分配。在Servlet/JSP容器中,一般僅實(shí)例化一個(gè)Servlet/JSP實(shí)例,啟動(dòng)多個(gè)該實(shí)例的線(xiàn)程來(lái)處理請求。而實(shí)例變量是該實(shí)例所有的線(xiàn)程所共享,所以,實(shí)例變量不是線(xiàn)程安全的。
3. 局部變量
局部變量在堆棧中分配,因為每一個(gè)線(xiàn)程有自己的執行堆棧,所以,局部變量是線(xiàn)程安全的。
二、在Servlet/JSP中的多線(xiàn)程同步問(wèn)題
在JSP中,使用實(shí)例變量要特別謹慎。首先請看下面的代碼:
// instanceconcurrenttest.jsp
<%@ page contentType="text/html;charset=GBK" %>
<%!
//定義實(shí)例變量
String username;
String password;
java.io.PrintWriter output;
%>
<%
//從request中獲取參數
username = request.getParameter("username");
password = request.getParameter("password");
output = response.getWriter();
showUserInfo();
%>
<%!
public void showUserInfo() {
//為了突出并發(fā)問(wèn)題,在這兒首先執行一個(gè)費時(shí)操作
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>");
}
%>
在這個(gè)頁(yè)面中,首先定義了兩個(gè)實(shí)例變量,username和password。然后在從request中獲取這兩個(gè)參數,并調用showUserInfo()方法將請求用戶(hù)的信息回顯在該客戶(hù)的瀏覽器上。在一個(gè)用戶(hù)訪(fǎng)問(wèn)是,不存在問(wèn)題。但在多個(gè)用戶(hù)并發(fā)訪(fǎng)問(wèn)時(shí),就會(huì )出現其它用戶(hù)的信息顯示在另外一些用戶(hù)的瀏覽器上的問(wèn)題。這是一個(gè)嚴重的問(wèn)題。為了突出并發(fā)問(wèn)題,便于測試、觀(guān)察,我們在回顯用戶(hù)信息時(shí)執行了一個(gè)模擬的費時(shí)操作,比如,下面的兩個(gè)用戶(hù)同時(shí)訪(fǎng)問(wèn)(可以啟動(dòng)兩個(gè)IE瀏覽器,或者在兩臺機器上同時(shí)訪(fǎng)問(wèn)):
a: http://localhost:8080/instanceconcurrenttest.jsp?username=a&password=123
b: http://localhost:8080/instanceconcurrenttest.jsp?username=b&password=456
如果a點(diǎn)擊鏈接后,b再點(diǎn)擊鏈接,那么,a將返回一個(gè)空白屏幕,b則得到a以及b兩個(gè)線(xiàn)程的輸出。請看下面的屏幕截圖:
圖1:a的屏幕
圖2:b的屏幕
從運行結果的截圖上可以看到,Web服務(wù)器啟動(dòng)了兩個(gè)線(xiàn)程分別來(lái)處理來(lái)自a和b的請求,但是在a卻得到一個(gè)空白的屏幕。這是因為上面程序中的output, username和password都是實(shí)例變量,是所有線(xiàn)程共享的。在a訪(fǎng)問(wèn)該頁(yè)面后,將output設置為a的輸出,username,password分別置為a的信息,而在a執行printUserInfo()輸出username和password信息前,b又訪(fǎng)問(wèn)了該頁(yè)面,把username和password置為了b的信息,并把輸出output指向到了b。隨后a的線(xiàn)程打印時(shí),就打印到了b的屏幕了,并且,a的用戶(hù)名和密碼也被b的取代。請參加下圖所示:
圖3:a、b兩個(gè)線(xiàn)程的時(shí)間線(xiàn)
而實(shí)際程序中,由于設置實(shí)例變量,使用實(shí)例變量這兩個(gè)時(shí)間點(diǎn)非常接近,所以,像本例的同步問(wèn)題并沒(méi)有這么突出,可能會(huì )偶爾出現,但這卻更加具有危險性,也更加難于調試。
同樣,對于Servlet也存在實(shí)例變量的多線(xiàn)程問(wèn)題,請看上面頁(yè)面的Servlet版:
// InstanceConcurrentTest.java
import 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() {
//為了突出并發(fā)問(wèn)題,在這兒首先執行一個(gè)費時(shí)操作
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. 以單線(xiàn)程運行Servlet/JSP
在JSP中,通過(guò)設置:,在Servlet中,通過(guò)實(shí)現javax.servlet.SingleThreadModel,此時(shí)Web容器將保證JSP或Servlet實(shí)例以單線(xiàn)程方式運行。
重要提示:在測試中發(fā)現,Tomcat 4.1.17不能正確支持isThreadSafe屬性,所以,指定isTheadSafe為false后,在Tomcat 4.1.17中仍然出現多線(xiàn)程問(wèn)題,這是Tomcat 4.1.17的Bug。在Tomcat 3.3.1和Resin 2.1.5中測試通過(guò)。
2. 去除實(shí)例變量,通過(guò)參數傳遞
從上面的分析可見(jiàn),應該在Servlet/JSP中盡量避免使用實(shí)例變量。比如,下面的修正代碼,去除了實(shí)例變量,通過(guò)定義局部變量,并參數進(jìn)行傳遞。這樣,由于局部變量是在線(xiàn)程的堆棧中進(jìn)行分配的,所以是線(xiàn)程安全的。不會(huì )出現多線(xiàn)程同步的問(wèn)題。代碼如下:
<%@ 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) {
//為了突出并發(fā)問(wèn)題,在這兒首先執行一個(gè)費時(shí)操作
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()方法或者實(shí)例變量的相關(guān)操作語(yǔ)句上使用synchronized關(guān)鍵字進(jìn)行同步,但這樣并不能解決多線(xiàn)程的問(wèn)題。因為,這樣雖然可以使對實(shí)例變量的操作代碼進(jìn)行同步,但并不能阻止一個(gè)線(xiàn)程使用另外一個(gè)線(xiàn)程修改后的“臟的”實(shí)例變量。所以,除了降低運行效率外,不會(huì )起到預期效果。
本站僅提供存儲服務(wù),所有內容均由用戶(hù)發(fā)布,如發(fā)現有害或侵權內容,請
點(diǎn)擊舉報。