Java如何在臨界區(qū)中避免競態(tài)條件
當兩個線程競爭同一資源時,如果對資源的訪問順序敏感,就稱存在競態(tài)條件。導致競態(tài)條件發(fā)生的代碼區(qū)稱作臨界區(qū)。在臨界區(qū)中使用適當?shù)耐骄涂梢员苊飧倯B(tài)條件。
界區(qū)實現(xiàn)方法有兩種,一種是用synchronized,一種是用Lock顯式鎖實現(xiàn)。synchronized 關鍵字,代表這個方法加鎖,相當于不管哪一個線程(例如線程A),運行到這個方法時,都要檢查有沒有其它線程B(或者C、 D等)正在用這個方法(或者該類的其他同步方法),有的話要等正在使用synchronized方法的線程B(或者C 、D)運行完這個方法后再運行此線程A,沒有的話,鎖定調用者,然后直接運行。它包括兩種用法:synchronized 方法和 synchronized 塊。
Java語言的關鍵字,可用來給對象和方法或者代碼塊加鎖,當它鎖定一個方法或者一個代碼塊的時候,同一時刻最多只有一個線程執(zhí)行這段代碼。當兩個并發(fā)線程訪問同一個對象object中的這個加鎖同步代碼塊時,一個時間內只能有一個線程得到執(zhí)行。另一個線程必須等待當前線程執(zhí)行完這個代碼塊以后才能執(zhí)行該代碼塊。然而,當一個線程訪問object的一個加鎖代碼塊時,另一個線程仍可以訪問該object中的非加鎖代碼塊。
有臨界區(qū)是為了讓更多的其它線程能安全夠訪問資源。
class Counter { protected long count = 0; public void add(long value) { this.count = this.count + value; } }
觀察線程A和B交錯執(zhí)行會發(fā)生什么,兩個線程分別加了2和3到count變量上,兩個線程執(zhí)行結束后count變量的值應該等于5。然而由于兩個線程是交叉執(zhí)行的,兩個線程從內存中讀出的初始值都是0。然后各自加了2和3,并分別寫回內存。最終的值并不是期望的5,而是最后寫回內存的那個線程的值,上面例子中最后寫回內存的是線程A,但實際中也可能是線程B。如果沒有采用合適的同步機制,線程間的交叉執(zhí)行情況就無法預料。
add()方法就是一個臨界區(qū),它會產(chǎn)生競態(tài)條件。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關文章:
