<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    自由飛翔

    我在仰望,java之上

    統計

    留言簿(2)

    我關注的blog

    閱讀排行榜

    評論排行榜

    JSP、servlet、struts線程安全問題分析

    Servlet之所以比CGI效率高就是因為Servlet是多線程的。
    Servlet規范已經聲明Servlet不是線程安全的.
    JSP中使用聲明的變量是Servlet的實例變量,不是線程安全的,其他都是線程安全的。

    那么怎樣才能是Servlet安全呢,凡是多個線程可以共享的就不要使用(實例變量+類變量),就這么簡單。也可以使用synchronized同步方法,但是這樣效率不高,還可以使用單線程模型,這樣的話效率就更低了,100個請求同時來的時候就要實例化100個實例。

    方法中的臨時變量是不會影響線程安全的,因為他們是在棧上分配空間,而且每個線程都有自己私有的棧空間。

    總結:線程安全問題主要是由實例變量造成的,不管在Servlet還是JSP,或者在Struts的Action里面,不要使用實例變量,
    任何方法里面都不要出現實例變量,你的程序就是線程安全的。


    在Servlet/JSP中的幾種變量類型
    源自 : http://www.javaresearch.org/article/8465.htm
    一、在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中,使用實例變量要特別謹慎。首先請看下面的代碼:
    // 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兩個線程的輸出。請看下面的屏幕截圖:
     
    從運行結果的截圖上可以看到,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的取代。請參加下圖所示:
    而實際程序中,由于設置實例變量,使用實例變量這兩個時間點非常接近,
    所以,像本例的同步問題并沒有這么突出,可能會偶爾出現,但這卻更加具有危險性,也更加難于調試。
    同樣,對于Servlet也存在實例變量的多線程問題,請看上面頁面的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() {
            //為了突出并發問題,在這兒首先執行一個費時操作
            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關鍵字進行同步,
    但這樣并不能解決多線程的問題。因為,這樣雖然可以使對實例變量的操作代碼進行同步,
    但并不能阻止一個線程使用另外一個線程修改后的“臟的”實例變量。
    所以,除了降低運行效率外,不會起到預期效果。


    待續...............


    Gavin

    posted on 2011-09-02 10:30 GavinMiao 閱讀(1079) 評論(0)  編輯  收藏 所屬分類: corejava

    主站蜘蛛池模板: 97se亚洲国产综合自在线| 无码久久精品国产亚洲Av影片| 一本色道久久88—综合亚洲精品 | 四虎永久在线精品免费影视| 最新国产成人亚洲精品影院| 成年女人毛片免费播放人| 亚洲成a人片在线观看精品| 黄瓜视频影院在线观看免费| 亚洲中文字幕无码久久2020| 亚洲日本视频在线观看| 亚洲免费人成视频观看| 亚洲成人激情小说| 亚洲免费在线播放| 亚洲国产成人久久| 拍拍拍又黄又爽无挡视频免费| 亚洲理论精品午夜电影| 亚色九九九全国免费视频| 亚洲人成影院77777| 成人最新午夜免费视频| 视频一区二区三区免费观看| 亚洲国产精品成人| 国产猛男猛女超爽免费视频| 亚洲综合一区二区| 午夜成人免费视频| 久久www免费人成精品香蕉| 亚洲AV无码乱码在线观看富二代 | 国内精自视频品线六区免费| 亚洲午夜无码毛片av久久京东热| 国产精品嫩草影院免费| aa毛片免费全部播放完整| 亚洲黄色高清视频| 精品免费国产一区二区| eeuss免费影院| 中文字幕亚洲精品| 日本高清免费中文字幕不卡| 中文毛片无遮挡高清免费| 亚洲国产午夜精品理论片| 亚洲成a人在线看天堂无码| 最近免费字幕中文大全视频| 亚洲AV永久无码天堂影院 | 永久黄色免费网站|