一、簡(jiǎn)介
首先讓我們來(lái)了解下在操作系統(tǒng)中進(jìn)程和線程的區(qū)別:
進(jìn)程:每個(gè)進(jìn)程都有獨(dú)立的代碼和數(shù)據(jù)空間(進(jìn)程上下文),進(jìn)程間的切換會(huì)有較大的開(kāi)銷(xiāo),一個(gè)進(jìn)程包含1--n個(gè)線程。(進(jìn)程是資源分配的最小單位)
線程:同一類(lèi)線程共享代碼和數(shù)據(jù)空間,每個(gè)線程有獨(dú)立的運(yùn)行棧和程序計(jì)數(shù)器(PC),線程切換開(kāi)銷(xiāo)小。(線程是cpu調(diào)度的最小單位)
線程和進(jìn)程一樣分為五個(gè)階段:創(chuàng)建、就緒、運(yùn)行、阻塞、終止。
多進(jìn)程是指操作系統(tǒng)能同時(shí)運(yùn)行多個(gè)任務(wù)(程序)。
多線程是指在同一程序中有多個(gè)順序流在執(zhí)行。
二、實(shí)現(xiàn)方式
(1)繼承Thread類(lèi),重寫(xiě)run函數(shù)?
創(chuàng)建:?
class xx extends Thread{?
public void run(){?
Thread.sleep(1000)//線程休眠1000毫秒,sleep使線程進(jìn)入Block狀態(tài),并釋放資源?
}}?
開(kāi)啟線程:?
對(duì)象.start()//啟動(dòng)線程,run函數(shù)運(yùn)行 。
注意:start()方法的調(diào)用后并不是立即執(zhí)行多線程代碼,而是使得該線程變?yōu)榭蛇\(yùn)行態(tài)(Runnable),什么時(shí)候運(yùn)行是由操作系統(tǒng)決定的。
從程序運(yùn)行的結(jié)果可以發(fā)現(xiàn),多線程程序是亂序執(zhí)行。因此,只有亂序執(zhí)行的代碼才有必要設(shè)計(jì)為多線程。
Thread.sleep()方法調(diào)用目的是不讓當(dāng)前線程獨(dú)自霸占該進(jìn)程所獲取的CPU資源,以留出一定時(shí)間給其他線程執(zhí)行的機(jī)會(huì)。
實(shí)際上所有的多線程代碼執(zhí)行順序都是不確定的,每次執(zhí)行的結(jié)果都是隨機(jī)的。
(2)實(shí)現(xiàn)Runnable接口,重寫(xiě)run函數(shù)?
開(kāi)啟線程:?
Thread t = new Thread(對(duì)象)//創(chuàng)建線程對(duì)象?
t.start()?
(3)實(shí)現(xiàn)Callable接口,重寫(xiě)call函數(shù)?
Callable是類(lèi)似于Runnable的接口,實(shí)現(xiàn)Callable接口的類(lèi)和實(shí)現(xiàn)Runnable的類(lèi)都是可被其它線程執(zhí)行的任務(wù),該方法有返回結(jié)果。?
Callable和Runnable有幾點(diǎn)不同:?
a、Callable規(guī)定的方法是call(),而Runnable規(guī)定的方法是run().?
b、Callable的任務(wù)執(zhí)行后可返回值,而Runnable的任務(wù)是不能返回值的?
c、call()方法可拋出異常,而run()方法是不能拋出異常的。?
d、運(yùn)行Callable任務(wù)可拿到一個(gè)Future對(duì)象,F(xiàn)uture表示異步計(jì)算的結(jié)果。它提供了檢查計(jì)算是否完成的方法,等待計(jì)算的完成,并檢索計(jì)算的結(jié)果.通過(guò)Future對(duì)象可了解任務(wù)執(zhí)行情況,可取消任務(wù)的執(zhí)行,還可獲取任務(wù)執(zhí)行的結(jié)果。