程序員大家都在用的并發(fā)到底是個(gè)什么_第1頁
程序員大家都在用的并發(fā)到底是個(gè)什么_第2頁
程序員大家都在用的并發(fā)到底是個(gè)什么_第3頁
程序員大家都在用的并發(fā)到底是個(gè)什么_第4頁
程序員大家都在用的并發(fā)到底是個(gè)什么_第5頁
已閱讀5頁,還剩5頁未讀, 繼續(xù)免費(fèi)閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報(bào)或認(rèn)領(lǐng)

文檔簡介

程序員大家都在用的并發(fā)到底是個(gè)什么?資料僅供參考寫在前面并發(fā)與操作系統(tǒng)的生命歷程息息相關(guān)。進(jìn)程的出現(xiàn),使得程序狀態(tài)的保存變?yōu)楝F(xiàn)實(shí),為進(jìn)程間的切換提供了可能,實(shí)現(xiàn)了操作系統(tǒng)的并發(fā),大大提高資源利用率。雖然進(jìn)程的出現(xiàn)解決了操作系統(tǒng)的并發(fā)問題,但人們對實(shí)時(shí)性又有了更高的要求。由于一個(gè)進(jìn)程由若干個(gè)子任務(wù)組成,因此人們就創(chuàng)造了線程,讓每個(gè)線程負(fù)責(zé)一個(gè)獨(dú)立的子任務(wù),提高程序的響應(yīng)靈敏度。一個(gè)進(jìn)程雖然包括多個(gè)線程,可是這些線程是共同享有進(jìn)程占有的資源和地址空間的。因此,雖然多線程提高了資源利用率,保證了實(shí)時(shí)性,但同時(shí)也帶來了包括安全性、活躍性和性能等問題。總的來說,進(jìn)程讓操作系統(tǒng)的并發(fā)性成為可能,而線程讓進(jìn)程的內(nèi)部并發(fā)成為可能。進(jìn)程和線程的由來操作系統(tǒng)中為什么會(huì)出現(xiàn)進(jìn)程?說起進(jìn)程的由來,我們需要從操作系統(tǒng)的發(fā)展歷史談起。可能在今天,我們無法想象在很多年以前計(jì)算機(jī)是什么樣子。我們現(xiàn)在能夠用計(jì)算機(jī)來做很多事情:辦公、娛樂、上網(wǎng),可是在計(jì)算機(jī)剛出現(xiàn)的時(shí)候,是為了解決數(shù)學(xué)計(jì)算的問題,因?yàn)楹芏啻罅康挠?jì)算經(jīng)過人力去完成是很耗時(shí)間和人力成本的。在最初的時(shí)候,計(jì)算機(jī)只能接受一些特定的指令,用戶輸入一個(gè)指令,計(jì)算機(jī)就做一個(gè)操作。當(dāng)用戶在思考或者輸入數(shù)據(jù)時(shí),計(jì)算機(jī)就在等待。顯然,這樣效率會(huì)很低下,因?yàn)楹芏鄷r(shí)候,計(jì)算機(jī)處于等待用戶輸入的狀態(tài)。那么,能不能把一系列需要操作的指令預(yù)先寫下來,形成一個(gè)清單,然后一次性交給計(jì)算機(jī),計(jì)算機(jī)不斷地去讀取指令來進(jìn)行相應(yīng)的操作?就這樣,

批處理操作系統(tǒng)誕生了。用戶能夠?qū)⑿枰獔?zhí)行的多個(gè)程序?qū)懺诖艓?,然后交由?jì)算機(jī)去讀取并逐個(gè)地執(zhí)行這些程序,并將輸出結(jié)果寫到另一個(gè)磁帶上。雖然批處理操作系統(tǒng)的誕生極大地提高了任務(wù)處理的便捷性,可是依然存在一個(gè)很大的問題:假如有兩個(gè)任務(wù)A和B,任務(wù)A在執(zhí)行到一半的過程中,需要讀取大量的數(shù)據(jù)輸入(I/O操作),而此時(shí)CPU只能靜靜地等待任務(wù)A讀取完數(shù)據(jù)才能繼續(xù)執(zhí)行,這樣就白白浪費(fèi)了CPU資源。人們于是想,能否在任務(wù)A讀取數(shù)據(jù)的過程中,讓任務(wù)B去執(zhí)行,當(dāng)任務(wù)A讀取完數(shù)據(jù)之后,讓任務(wù)B暫停,然后讓任務(wù)A繼續(xù)執(zhí)行?可是這樣就有一個(gè)問題,原來每次都是一個(gè)程序在計(jì)算機(jī)里面運(yùn)行,也就說內(nèi)存中始終只有一個(gè)程序的運(yùn)行數(shù)據(jù)。而如果想要任務(wù)A執(zhí)行I/O操作的時(shí)候,讓任務(wù)B去執(zhí)行,必然內(nèi)存中要裝入多個(gè)程序,那么如何處理呢?多個(gè)程序使用的數(shù)據(jù)如何進(jìn)行辨別呢?而且,當(dāng)一個(gè)程序運(yùn)行暫停后,后面如何恢復(fù)到它之前執(zhí)行的狀態(tài)呢?這個(gè)時(shí)候,人們就創(chuàng)造了進(jìn)程,用進(jìn)程來對應(yīng)一個(gè)程序,每個(gè)進(jìn)程對應(yīng)一定的內(nèi)存地址空間,而且只能使用它自己的內(nèi)存空間,各個(gè)進(jìn)程間互不干擾。而且,進(jìn)程保存了程序每個(gè)時(shí)刻的運(yùn)行狀態(tài),這樣就為進(jìn)程切換提供了可能。當(dāng)進(jìn)程暫停時(shí),它會(huì)保存當(dāng)前進(jìn)程的狀態(tài)(比如進(jìn)程標(biāo)識、進(jìn)程的使用的資源等),在下一次重新切換回來時(shí),便根據(jù)之前保存的狀態(tài)進(jìn)行恢復(fù),然后繼續(xù)執(zhí)行。這就是并發(fā),能夠讓操作系統(tǒng)從宏觀上看起來同一個(gè)時(shí)間段有多個(gè)任務(wù)在執(zhí)行。換句話說,進(jìn)程讓操作系統(tǒng)的并發(fā)成為了可能。注意,雖然并發(fā)從宏觀上看有多個(gè)任務(wù)在執(zhí)行,可是事實(shí)上,任一個(gè)具體的時(shí)刻,只有一個(gè)任務(wù)在占用CPU資源(當(dāng)然是對于單核CPU來說的)。為什么會(huì)出現(xiàn)線程?在出現(xiàn)了進(jìn)程以后,操作系統(tǒng)的性能得到了大大的提升。雖然進(jìn)程的出現(xiàn)解決了操作系統(tǒng)的并發(fā)問題,可是人們依然不滿足,人們逐漸對實(shí)時(shí)性有了要求。因?yàn)橐粋€(gè)進(jìn)程在一個(gè)時(shí)間段內(nèi)只能做一件事情,如果一個(gè)進(jìn)程有多個(gè)子任務(wù),只能逐個(gè)地去執(zhí)行這些子任務(wù)。比如,對于一個(gè)監(jiān)控系統(tǒng)來說,它不但要把圖像數(shù)據(jù)顯示在畫面上,還要與服務(wù)端進(jìn)行通信獲取圖像數(shù)據(jù),還要處理人們的交互操作。如果某一個(gè)時(shí)刻該系統(tǒng)正在與服務(wù)器通信獲取圖像數(shù)據(jù),而用戶又在監(jiān)控系統(tǒng)上點(diǎn)擊了某個(gè)按鈕,那么該系統(tǒng)就要等待獲取完圖像數(shù)據(jù)之后才能處理用戶的操作,如果獲取圖像數(shù)據(jù)需要耗費(fèi)10s,那么用戶就只有一直等待。顯然,對于這樣的系統(tǒng),人們是無法滿足的。那么,可不能夠?qū)⑦@些子任務(wù)分開執(zhí)行呢?即,在系統(tǒng)獲取圖像數(shù)據(jù)的同時(shí),如果用戶點(diǎn)擊了某個(gè)按鈕,則會(huì)暫停獲取圖像數(shù)據(jù),而先去響應(yīng)用戶的操作(因?yàn)橛脩舻牟僮魍鶊?zhí)行時(shí)間很短),在處理完用戶操作之后,再繼續(xù)獲取圖像數(shù)據(jù)。人們就創(chuàng)造了線程,讓一個(gè)線程去執(zhí)行一個(gè)子任務(wù),這樣一個(gè)進(jìn)程就包括了多個(gè)線程,每個(gè)線程負(fù)責(zé)一個(gè)獨(dú)立的子任務(wù)。這樣,在用戶點(diǎn)擊按鈕的時(shí)候,就能夠暫停獲取圖像數(shù)據(jù)的線程,讓UI線程響應(yīng)用戶的操作,響應(yīng)完之后再切換回來,讓獲取圖像的線程得到CPU資源。從而,讓用戶感覺系統(tǒng)是同時(shí)在做多件事情的,滿足了用戶對實(shí)時(shí)性的要求。換句話說,進(jìn)程讓操作系統(tǒng)的并發(fā)性成為可能,而線程讓進(jìn)程的內(nèi)部并發(fā)成為可能??墒且⒁?,一個(gè)進(jìn)程雖然包括多個(gè)線程,可是這些線程是共同享有進(jìn)程占有的資源和地址空間的。進(jìn)程是操作系統(tǒng)進(jìn)行資源分配的基本單位,而線程是操作系統(tǒng)進(jìn)行調(diào)度的基本單位。多線程并發(fā)由于多個(gè)線程是共同占有所屬進(jìn)程的資源和地址空間的,那么就會(huì)存在一個(gè)問題:如果多個(gè)線程要同時(shí)訪問某個(gè)資源,怎么處理?這個(gè)問題就是并發(fā)安全性問題。另外,可能有朋友會(huì)問,現(xiàn)在很多時(shí)候都采用多線程編程,那么是不是多線程的性能一定就由于單線程呢?答案是不一定,要看具體的任務(wù)以及計(jì)算機(jī)的配置。比如說:對于單核CPU,如果是CPU密集型任務(wù),如解壓文件,多線程的性能反而不如單線程性能,因?yàn)榻鈮何募枰恢闭加肅PU資源,如果采用多線程,線程切換導(dǎo)致的開銷反而會(huì)讓性能下降??墒菍τ诒热缃换ヮ愋偷娜蝿?wù),肯定是需要使用多線程的。而對于多核CPU,對于解壓文件來說,多線程肯定優(yōu)于單線程,因?yàn)槎鄠€(gè)線程能夠更加充分利用每個(gè)核的資源。雖然多線程能夠提升程序性能,可是相對于單線程來說,它的編程要復(fù)雜地多,要考慮線程安全問題。因此,在實(shí)際編程過程中,要根據(jù)實(shí)際情況具體選擇。并發(fā)簡史總結(jié)早期的計(jì)算機(jī)不包含操作系統(tǒng),它們從頭到尾只執(zhí)行一個(gè)程序,而且這個(gè)程序能夠訪問計(jì)算機(jī)中的所有資源。這對于昂貴且稀有的計(jì)算機(jī)資源來說是一種浪費(fèi);操作系統(tǒng)的出現(xiàn)使得計(jì)算機(jī)能同時(shí)運(yùn)行多個(gè)程序,不同的程序都在單獨(dú)的進(jìn)程中運(yùn)行,而且操作系統(tǒng)為各個(gè)獨(dú)立執(zhí)行的進(jìn)程分配資源(eg:經(jīng)過粗粒度時(shí)間分片使程序共享資源,如CPU等)。這無疑提高了計(jì)算機(jī)資源的利用率;在早期的分時(shí)系統(tǒng)中,每個(gè)進(jìn)程的執(zhí)行都是串行的。串行編程模型的優(yōu)勢在于其簡單性和直觀性,因?yàn)樗看沃蛔鲆患虑?,做完之后再做另一件。這種串行編程模型依然存在著計(jì)算機(jī)資源利用率不高的問題;促使進(jìn)程出現(xiàn)的因素同樣也促使著線程的出現(xiàn)。線程允許在同一個(gè)進(jìn)程中同時(shí)存在多個(gè)程序控制流。線程會(huì)共享進(jìn)程范圍內(nèi)的資源,但每個(gè)線程都有各自的程序計(jì)數(shù)器、棧以及局部變量等等;線程也被成為輕量級進(jìn)程。在大多數(shù)現(xiàn)代操作系統(tǒng)中,都是以線程為基本的調(diào)度單位,而不是進(jìn)程。如果沒有明確的協(xié)同機(jī)制,那么線程將彼此獨(dú)立執(zhí)行。由于同一個(gè)進(jìn)程的所有線程都將共享進(jìn)程的內(nèi)存地址空間,因此這些線程都能訪問相同的變量,這就需要實(shí)現(xiàn)一種比進(jìn)程間共享數(shù)據(jù)粒度更細(xì)的數(shù)據(jù)共享機(jī)制。如果沒有明確的同步機(jī)制來協(xié)同對共享數(shù)據(jù)的訪問,將造成不可預(yù)測的結(jié)果。線程的優(yōu)勢解耦、簡化程序開發(fā)在程序中,如果我們?yōu)槊糠N類型的任務(wù)都分配一個(gè)專門的線程,那么能夠形成一種串行執(zhí)行的假象,并將程序的執(zhí)行邏輯與調(diào)度機(jī)制的細(xì)節(jié),交替執(zhí)行的操作,異步I/O以及資源等待等問題分離開來。經(jīng)過使用線程,能夠?qū)?fù)雜而且異步的工作流進(jìn)一步分解為一組簡單而且同步的工作流,每個(gè)工作流在一個(gè)單獨(dú)的線程中運(yùn)行,并在特定的同步位置進(jìn)行交互。Servlet框架就是一個(gè)很好的例子??蚣茇?fù)責(zé)解決一些細(xì)節(jié)問題,包括請求管理、線程創(chuàng)立、負(fù)載平衡等,并在正確的時(shí)刻將請求分發(fā)給正確的應(yīng)用程序組件(對應(yīng)的一個(gè)具體Servlet)。編寫Servlet的開發(fā)人員不需要了解有多少請求在同一時(shí)刻被處理,也不需要了解套接字的輸入(出)流是否被阻塞。當(dāng)調(diào)用Servlet的service方法來響應(yīng)Web請求時(shí),能夠以同步方式來處理這個(gè)請求,就仿佛它是一個(gè)單線程的程序。這種方式簡化了組件的開發(fā),大大降低框架學(xué)習(xí)門檻。多線程還有助于用戶界面的靈敏響應(yīng)。例如,在Android開發(fā)中,我們常常將網(wǎng)路請求或I/O等耗時(shí)操作單獨(dú)放到一個(gè)線程中,以提高響應(yīng)的靈敏度。提高資源利用率多處理器系統(tǒng)的出現(xiàn),使得同一個(gè)程序的多個(gè)線程能夠被同時(shí)調(diào)度到多個(gè)CPU上運(yùn)行。因此,多線程程序能夠經(jīng)過提高處理器資源的利用率來提升系統(tǒng)的吞吐率。其實(shí),多線程程序也有助于在單處理器系統(tǒng)上獲得更高的吞吐率(如果程序的一個(gè)線程在等待I/O操作的完成,另一個(gè)線程能夠繼續(xù)運(yùn)行,使程序能夠在I/O阻塞期間繼續(xù)運(yùn)行)。線程帶來的風(fēng)險(xiǎn)安全性問題在線程安全性的定義中,最核心的概念就是正確性。當(dāng)多個(gè)線程訪問某個(gè)類時(shí),不論運(yùn)行時(shí)環(huán)境采用何種調(diào)度方式或者這些線程將如何交替執(zhí)行,而且在主調(diào)代碼中不需要任何額外的同步或協(xié)同,這個(gè)類都能表現(xiàn)出正確的行為,那么這個(gè)類就是線程安全的。線程不安全類示例:@NotThreadSafe

publicclassUnsafeSequence{

privateintvalue;

/**Returnsauniquevalue.*/

publicintgetNext(){

returnvalue++;

}

}雖然

遞增運(yùn)算“value++”看上去是單個(gè)操作,但實(shí)際上它包含三個(gè)獨(dú)立的操作:讀取value,將value加1,并將計(jì)算結(jié)果寫入value。由于運(yùn)行時(shí)各個(gè)線程執(zhí)行順序的不確定性,可能這段代碼在不同線程的調(diào)用中返回相同的數(shù)值活躍性問題活躍性問題關(guān)注的是:某件正確的事情最終會(huì)發(fā)生。導(dǎo)致活躍性的問題包括死鎖、饑餓等。性能問題性能問題關(guān)注的是:正確的事情能夠盡快發(fā)生。性能問題包括多個(gè)方面,例如響應(yīng)不靈敏,吞吐率過低,資源消耗過高等。在多線程程序中,當(dāng)線程調(diào)度器臨時(shí)掛起活躍線程并轉(zhuǎn)而運(yùn)行另一個(gè)線程時(shí),就會(huì)頻繁出現(xiàn)上下文切換操作(ContextSwitch),這種操作會(huì)導(dǎo)致CPU時(shí)間更多的花在線程調(diào)度上而非線程的運(yùn)行上。線程無處不在在Java中,一個(gè)應(yīng)用程序?qū)?yīng)著一個(gè)JVM實(shí)例(JVM進(jìn)程)。Java采用的是單線程編程模型,即在我們自己的程序中如果沒有主動(dòng)創(chuàng)立線程的話,只會(huì)創(chuàng)立一個(gè)線程,一般稱為主線程??墒且⒁?,雖然只有一個(gè)線程來執(zhí)行任務(wù),不代表JVM中只有一個(gè)線程,JVM實(shí)例在創(chuàng)立的時(shí)候,同時(shí)會(huì)創(chuàng)立很多其它的線程(比如垃圾收集器線程)。由于Java采用的是單線程編程模型,因此在進(jìn)行UI編程時(shí)要注意將耗時(shí)的操作放在子線程中進(jìn)行,以避免阻塞主線程(在UI編程時(shí),主線程即UI線程,用來處理用戶的交互事件)。publicclassTest{

publicstaticvoidmain(String[]args)

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論