進(jìn)程與線程的區(qū)別
進(jìn)程是指正在運(yùn)行的程序,幾乎所有的操作系統(tǒng)都支持進(jìn)程的概念,所有運(yùn)行的任務(wù)通常對(duì)應(yīng)一個(gè)進(jìn)程,當(dāng)一個(gè)程序進(jìn)入內(nèi)存執(zhí)行田弥,即變成了一個(gè)進(jìn)程,進(jìn)程是處于運(yùn)行中的程序铡原,并且具有一定的獨(dú)立功能偷厦,進(jìn)程是系統(tǒng)進(jìn)行資源分配和調(diào)度的一個(gè)獨(dú)立單位。線程是進(jìn)程的執(zhí)行單元眷蜈,一個(gè)進(jìn)程可以擁有多個(gè)線程沪哺,多個(gè)線程之間共享進(jìn)程所擁有的全部資源。一個(gè)進(jìn)程中的多個(gè)線程可以并發(fā)執(zhí)行酌儒,線程的執(zhí)行是搶占式的辜妓,線程的調(diào)度和管理由進(jìn)程本身負(fù)責(zé)完成,
進(jìn)程之間不能共享內(nèi)存忌怎,但線程之間共享內(nèi)存很容易籍滴,
系統(tǒng)創(chuàng)建進(jìn)程是需要為進(jìn)程分配系統(tǒng)資源,但創(chuàng)建線程則代價(jià)小得多
java中創(chuàng)建線程的方式
- 通過(guò)繼承Thread類(lèi)
- 定義繼承Thread類(lèi)的子類(lèi)榴啸,重寫(xiě)run()方法孽惰,run方法即為線程執(zhí)行體
- 創(chuàng)建該子類(lèi)的實(shí)例對(duì)象,即創(chuàng)建了線程對(duì)象
- 調(diào)用該對(duì)象的start()方法啟動(dòng)執(zhí)行該線程
- 通過(guò)實(shí)行Runnable接口
- 定義實(shí)現(xiàn)Runnable接口的類(lèi)鸥印,同樣實(shí)現(xiàn)run()方法
- 創(chuàng)建該類(lèi)的實(shí)例對(duì)象勋功,以該對(duì)象作為T(mén)hread的target創(chuàng)建Thread實(shí)例對(duì)象,該Thread對(duì)象才是真正的線程對(duì)象
- 調(diào)用該Thread對(duì)象的start() 方法啟動(dòng)線程
- 使用callable接口和FutureTask類(lèi)
- 創(chuàng)建Callbale接口的實(shí)現(xiàn)類(lèi)库说,并且實(shí)現(xiàn)call()方法狂鞋,并且該call方法有返回值
- 創(chuàng)建Callable的實(shí)例對(duì)象,并用FutureTask類(lèi)來(lái)包裝Callable對(duì)象潜的,該FutureTask對(duì)象封裝了callable對(duì)象的call方法的返回值
- 使用FutureTask對(duì)象作為T(mén)hread對(duì)象的target創(chuàng)建并啟動(dòng)線程
- 調(diào)用FutureTask對(duì)象的get方法來(lái)獲得子線程執(zhí)行結(jié)束后的返回值
線程創(chuàng)建方式對(duì)比
采用接口的方式創(chuàng)建線程打破了類(lèi)單繼承的局限骚揍,還可以繼承其他類(lèi),只此一點(diǎn) 推薦使用接口方式
同樣 采用接口實(shí)現(xiàn)方式多個(gè)進(jìn)程之間可以共享一個(gè)target對(duì)象啰挪,適合多個(gè)線程來(lái)處理同一份資源的情況
線程同步安全問(wèn)題
采用同步代碼塊synchornized(鎖定對(duì)象)
或者同步方法 public synchronized 返回值 方法名(參數(shù)){//code}
或者同步鎖(Lock) 定義一個(gè)鎖對(duì)象 在方法的開(kāi)頭加鎖信不,方法結(jié)尾 解鎖
線程睡眠方法sleep() 與線程讓步y(tǒng)ield()方法的不同
sleep方法會(huì)將線程進(jìn)入阻塞狀態(tài),直到經(jīng)過(guò)阻塞時(shí)間才會(huì)轉(zhuǎn)人就緒狀態(tài)嘲叔,而yield方法不會(huì)將線程阻塞,只是將線程強(qiáng)制進(jìn)入就緒狀態(tài)抽活,讓線程調(diào)度器重新調(diào)度一次硫戈,完全有可能某個(gè)線程調(diào)用的yield暫停之后,立即再次獲得處理器資源被執(zhí)行酌壕。
sleep方法會(huì)調(diào)用拋出異常所以要么捕捉該異常要么拋出異常掏愁,而調(diào)用yield方法沒(méi)有聲明拋出任何異常歇由。
線程通信可以借助于Object類(lèi)的wait() notify() 和 notifyAll()方法
wait() 方法導(dǎo)致當(dāng)前線程等待卵牍,直到其他線程調(diào)用該同步監(jiān)視器的notify()方法或notifyAll()方法喚醒該線程,調(diào)用wait方法會(huì)釋放對(duì)當(dāng)前同步監(jiān)視器的鎖定
notify() 喚醒在此同步監(jiān)視器上等待的單個(gè)線程沦泌,如果所有線程都在此同步監(jiān)視器上等待糊昙,則會(huì)喚醒其中一個(gè)線程,選擇是任意的谢谦,只有當(dāng)前線程放棄對(duì)同步監(jiān)視器的鎖定后(即調(diào)用wait()方法)释牺,才可以執(zhí)行被喚醒的線程
notifyAll() 喚醒在此同步監(jiān)視器上等待的所有線程,只有當(dāng)前線程放棄對(duì)同步監(jiān)視器的鎖定后回挽,才可以執(zhí)行被喚醒的線程
經(jīng)典的生產(chǎn)者消費(fèi)者例子
通過(guò)使用一個(gè)flag標(biāo)識(shí)判斷是否可以生產(chǎn) 是否可以消費(fèi)
還可以通過(guò)阻塞隊(duì)列控制線程通信 BlockingQueue
包裝線程不安全的集合
通過(guò)Collections提供的靜態(tài)方法將這些集合包裝成線程安全的集合没咙,比如ArrayList HashSet TreeSet HashMap TreeMap 等都是線程不安全的