一、run()和start() 這兩個方法應(yīng)該都比較熟悉,把需要并行處理的代碼放在run()方法中,start()方法啟動線程將自動調(diào)用 run()方法,這是由Java的內(nèi)存機制規(guī)定的。并且run()方法必須是public訪問權(quán)限,返回值類型為void. 二、關(guān)鍵字Synchronized 這個關(guān)鍵字用于保護共享數(shù)據(jù),當然前提是要分清哪些數(shù)據(jù)是共享數(shù)據(jù)。每個對象都有一個鎖標志,當一個線程訪問該對象時,被Synchronized修飾的數(shù)據(jù)將被“上鎖”,阻止其他線程訪問。當前線程訪問完這部分數(shù)據(jù)后釋放鎖標志,其他線程就可以訪問了。
以上這段程序中的 i 變量并不是共享數(shù)據(jù),也就是這里的Synchronized關(guān)鍵字并未起作用。因為t1,t2兩個線程是兩個對象(r1,r2)的線程。不同的對象其數(shù)據(jù)是不同的,所以r1和r2兩個對象的i變量是并不是共享數(shù)據(jù)。 當把代碼改成如下:Synchronized關(guān)鍵字才會起作用
三、sleep() 使當前線程(即調(diào)用該方法的線程)暫停執(zhí)行一段時間,讓其他線程有機會繼續(xù)執(zhí)行,但它并不釋放對象鎖。也就是如果有Synchronized同步塊,其他線程仍然不同訪問共享數(shù)據(jù)。注意該方法要捕獲異常 比如有兩個線程同時執(zhí)行(沒有Synchronized),一個線程優(yōu)先級為MAX_PRIORITY,另一個為MIN_PRIORITY,如果沒有Sleep()方法,只有高優(yōu)先級的線程執(zhí)行完成后,低優(yōu)先級的線程才能執(zhí)行;但當高優(yōu)先級的線程sleep(5000)后,低優(yōu)先級就有機會執(zhí)行了。 總之,sleep()可以使低優(yōu)先級的線程得到執(zhí)行的機會,當然也可以讓同優(yōu)先級、高優(yōu)先級的線程有執(zhí)行的機會。 四、join() join()方法使調(diào)用該方法的線程在此之前執(zhí)行完畢,也就是等待調(diào)用該方法的線程執(zhí)行完畢后再往下繼續(xù)執(zhí)行。注意該方法也要捕獲異常。 五、yield() 它與sleep()類似,只是不能由用戶指定暫停多長時間,并且yield()方法只能讓同優(yōu)先級的線程有執(zhí)行的機會。 六、wait()和notify()、notifyAll() 這三個方法用于協(xié)調(diào)多個線程對共享數(shù)據(jù)的存取,所以必須在Synchronized語句塊內(nèi)使用這三個方法。前面說過Synchronized 這個關(guān)鍵字用于保護共享數(shù)據(jù),阻止其他線程對共享數(shù)據(jù)的存取。但是這樣程序的流程就很不靈活了,如何才能在當前線程還沒退出Synchronized數(shù)據(jù)塊時讓其他線程也有機會訪問共享數(shù)據(jù)呢?此時就用這三個方法來靈活控制。 wait()方法使當前線程暫停執(zhí)行并釋放對象鎖標志,讓其他線程可以進入Synchronized數(shù)據(jù)塊,當前線程被放入對象等待池中。當調(diào)用 notify()方法后,將從對象的等待池中移走一個任意的線程并放到鎖標志等待池中,只有鎖標志等待池中的線程能夠獲取鎖標志;如果鎖標志等待池中沒有線程,則notify()不起作用。notifyAll()則從對象等待池中移走所有等待那個對象的線程并放到鎖標志等待池中。 |
|
來自: 々心寒々 > 《JAVA基礎(chǔ)知識》