posts - 101,  comments - 29,  trackbacks - 0
          先說明幾個(gè)概念:
          工作者線程Work Thread:執(zhí)行代碼的一組線程
          調(diào)度線程Dispatcher Thread:每個(gè)線程都具有分配給它的線程優(yōu)先級(jí),線程是根據(jù)優(yōu)先級(jí)調(diào)度執(zhí)行的
          Servlet采用多線程來處理多個(gè)請(qǐng)求同時(shí)訪問。servlet依賴于一個(gè)線程池來服務(wù)請(qǐng)求。線程池實(shí)際上是一系列的工作者線程集合。Servlet使用一個(gè)調(diào)度線程來管理工作者線程.
          當(dāng)容器收到一個(gè)Servlet請(qǐng)求,調(diào)度線程從線程池中選出一個(gè)工作者線程,將請(qǐng)求傳遞給該工作者線程,然后由該線程來執(zhí)行Servlet的service方法。當(dāng)這個(gè)線程正在執(zhí)行的時(shí)候,容器收到另外一個(gè)請(qǐng)求,調(diào)度線程同樣從線程池中選出另一個(gè)工作者線程來服務(wù)新的請(qǐng)求,容器并不關(guān)心這個(gè)請(qǐng)求是否訪問的是同一個(gè)Servlet.當(dāng)容器同時(shí)收到對(duì)同一個(gè)Servlet的多個(gè)請(qǐng)求的時(shí)候,那么這個(gè)Servlet的service()方法將在多線程中并發(fā)執(zhí)行。
          Servlet容器默認(rèn)采用單實(shí)例多線程的方式來處理請(qǐng)求,這樣減少產(chǎn)生Servlet實(shí)例的開銷,提升了對(duì)請(qǐng)求的響應(yīng)時(shí)間,對(duì)于Tomcat可以在server.xml中通過<Connector>元素設(shè)置線程池中線程的數(shù)目
          就實(shí)現(xiàn)來說:
          調(diào)度者線程類所擔(dān)負(fù)的責(zé)任如其名字,該類的責(zé)任是調(diào)度線程,只需要利用自己的屬性完成自己的責(zé)任。所以該類是承擔(dān)了責(zé)任的,并且該類的責(zé)任又集中到唯一的單體對(duì)象中。

          而其他對(duì)象又依賴于該特定對(duì)象所承擔(dān)的責(zé)任,我們就需要得到該特定對(duì)象。那該類就是一個(gè)單例模式的實(shí)現(xiàn)了。


          “servlet 可以同時(shí)處理多個(gè)請(qǐng)求”

          當(dāng)多個(gè)request同時(shí)來請(qǐng)求一個(gè)servlet時(shí),tomcat的工作原理是會(huì)對(duì)這多個(gè)請(qǐng)求分別創(chuàng)建線程

          但是每個(gè)線程拿到的servlet實(shí)例是同一個(gè)servlet實(shí)例(單例模式),這樣的話他們?cè)谑褂胹ervice方法時(shí)就會(huì)可能出現(xiàn)同時(shí)使用,所以如果有需要更改實(shí)例狀態(tài)(共享成員變量的)語句,就要加上鎖-synchronized關(guān)鍵字。


          1,變量的線程安全:這里的變量指字段和共享數(shù)據(jù)(如表單參數(shù)值)。

          a,將 參數(shù)變量 本地化。多線程并不共享局部變量.所以我們要盡可能的在servlet中使用局部變量。
          例如:String user = “”;
          user = request.getParameter(“user”);

          b,使用同步塊Synchronized,防止可能異步調(diào)用的代碼塊。這意味著線程需要排隊(duì)處理。
          在使用同板塊的時(shí)候要盡可能的縮小同步代碼的范圍,不要直接在sevice方法和響應(yīng)方法上使用同步,這樣會(huì)嚴(yán)重影響性能。

          2,屬性的線程安全:ServletContext,HttpSession,ServletRequest對(duì)象中屬性
          ServletContext:(線程是不安全的)
          ServletContext是可以多線程同時(shí)讀/寫屬性的,線程是不安全的。要對(duì)屬性的讀寫進(jìn)行同步處理或者進(jìn)行深度Clone()。
          所以在Servlet上下文中盡可能少量保存會(huì)被修改(寫)的數(shù)據(jù),可以采取其他方式在多個(gè)Servlet中共享,比方我們可以使用單例模式來處理共享數(shù)據(jù)。
          HttpSession:(線程是不安全的)
          HttpSession對(duì)象在用戶會(huì)話期間存在,只能在處理屬于同一個(gè)Session的請(qǐng)求的線程中被訪問,因此Session對(duì)象的屬性訪問理論上是線程安全的。
          當(dāng)用戶打開多個(gè)同屬于一個(gè)進(jìn)程的瀏覽器窗口,在這些窗口的訪問屬于同一個(gè)Session,會(huì)出現(xiàn)多次請(qǐng)求,需要多個(gè)工作線程來處理請(qǐng)求,可能造成同時(shí)多線程讀寫屬性。
          這時(shí)我們需要對(duì)屬性的讀寫進(jìn)行同步處理:使用同步塊Synchronized和使用讀/寫器來解決。

          ServletRequest:(線程是安全的)
          對(duì)于每一個(gè)請(qǐng)求,由一個(gè)工作線程來執(zhí)行,都會(huì)創(chuàng)建有一個(gè)新的ServletRequest對(duì)象,所以ServletRequest對(duì)象只能在一個(gè)線程中被訪問。ServletRequest是線程安全的。
          注意:ServletRequest對(duì)象在service方法的范圍內(nèi)是有效的,不要試圖在service方法結(jié)束后仍然保存請(qǐng)求對(duì)象的引用。

          3,使用同步的集合類:
          使用Vector代替ArrayList,使用Hashtable代替HashMap。

          4,不要在Servlet中創(chuàng)建自己的線程來完成某個(gè)功能。
          Servlet本身就是多線程的,在Servlet中再創(chuàng)建線程,將導(dǎo)致執(zhí)行情況復(fù)雜化,出現(xiàn)多線程安全問題。

          5,在多個(gè)servlet中對(duì)外部對(duì)象(比方文件)進(jìn)行修改操作一定要加鎖,做到互斥的訪問。

          6,javax.servlet.SingleThreadModel接口是一個(gè)標(biāo)識(shí)接口,如果一個(gè)Servlet實(shí)現(xiàn)了這個(gè)接口,那Servlet容器將保證在一個(gè)時(shí)刻僅有一個(gè)線程可以在給定的servlet實(shí)例的service方法中執(zhí)行。將其他所有請(qǐng)求進(jìn)行排隊(duì)。
          服務(wù)器可以使用多個(gè)實(shí)例來處理請(qǐng)求,代替單個(gè)實(shí)例的請(qǐng)求排隊(duì)帶來的效益問題。服務(wù)器創(chuàng)建一個(gè)Servlet類的多個(gè)Servlet實(shí)例組成的實(shí)例池,對(duì)于每個(gè)請(qǐng)求分配Servlet實(shí)例進(jìn)行響應(yīng)處理,之后放回到實(shí)例池中等待下此請(qǐng)求。這樣就造成并發(fā)訪問的問題。
          此時(shí),局部變量(字段)也是安全的,但對(duì)于
          全局變量和共享數(shù)據(jù)是不安全的,需要進(jìn)行同步處理。而對(duì)于這樣多實(shí)例的情況SingleThreadModel接口并不能解決并發(fā)訪問問題。
          java進(jìn)階 http://www.javady.com/index.php/category/hign_xingneng 



          posted on 2012-05-12 00:11 mixer-a 閱讀(2455) 評(píng)論(0)  編輯  收藏

          只有注冊(cè)用戶登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 洛隆县| 德化县| 龙山县| 碌曲县| 营口市| 庆阳市| 广水市| 如东县| 南岸区| 华亭县| 茶陵县| 乌什县| 宜兴市| 康平县| 法库县| 涿鹿县| 改则县| 五峰| 万山特区| 永州市| 法库县| 沾益县| 龙游县| 长阳| 郑州市| 泗洪县| 和平区| 龙州县| 吉林市| 龙川县| 行唐县| 镇赉县| 卫辉市| 大新县| 隆林| 镇原县| 卓尼县| 八宿县| 雅江县| 沙雅县| 石棉县|