天天看點

JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this

我們繼續上個篇幅接着講線程的知識點
當我們開啟四個視窗(線程)把票陸陸續續的賣完了之後,我們要反思一下,這裡面有沒有安全隐患呢?在實際情況中,這種事情我們是必須要去考慮安全問題的,那我們模拟一下錯誤
我們輸出的結果
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
這裡出現了0票,如果你繼續跟蹤的話,你會發現,還會出現-1,-2之類的票,這就是安全隐患,那原因是什麼呢?

當多條語句在操作同一個線程共享資料時,一個線程對多條語句隻執行了一個部分,還沒有執行完,另外一個線程參與了執行,導緻共享資料的錯誤

解決辦法:對多條操作共享資料的語句,隻能讓一個線程都執行完再執行過程中其他線程不可以參與運作 JAVA對多線程的安全問題提供了專業的解決辦法,就是同步代碼塊
那我們怎麼用呢?
這樣,就輸出
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
我們為什麼可以這樣去同步線程? 對象如同鎖,持有鎖的線程可以在同步中執行,沒有執行鎖的線程即使擷取了CPU的執行權,也進不去,因為沒有擷取鎖,我們可以這樣了解

四個線程,哪一個進去就開始執行,其他的拿不到執行權,是以即使拿到了執行權,也進不去,這個同步能解決線程的安全問題

但是,同步是有前提的

1.必須要有兩個或者兩個以上的線程,不然你同步也沒必要呀

2.必須是多個線程使用同一鎖

必須保證同步中隻能有一個線程在運作 但是他也有一個弊端:那就是多個線程都需要判斷鎖,較為消耗資源
我們可以寫一段小程式,來驗證這個線程同步的問題,也就是說我們看看下面這段程式是否有安全問題,有的話,如何解決?
當你執行的時候你會發現
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
這裡是沒錯的,存了600塊錢,但是,這個程式是有安全隐患的 如何找到問題?

1.明确哪些代碼是多線成運作代碼

2.明确共享資料

3.明确多線成運作代碼中哪些語句是操作共享資料的

那我們怎麼找到安全隐患呢?我們去銀行的類裡面做些認為操作
讓他sleep一下你就會發現
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
這樣的話,我們就可以使用我們的同步代碼了
這樣代碼就可以同步了
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
哪些代碼該同步,哪些不該同步,你一定要搞清楚,根據上面的3個條件 大家有沒有注意到,函數式具有封裝代碼的特定,而我們所操作的同步代碼塊也是有封裝代碼的特性,拿這樣的話我們就可以換一種形式去操作,那就是寫成函數的修飾符
這樣也是OK的
既然我們學習了另一種同步函數的寫法,那我們就可以把剛才的買票小例子進一步封裝一下了
但是這樣做,你卻會發現一個很嚴重的問題,那就是
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
永遠隻有0線程在執行賣票 那是因為我們并沒有搞清楚需要同步哪一個代碼段,我們應該執行的隻是裡面的那兩段代碼,而不是整個死循環,是以我們得封裝個函數進行線程同步
這樣輸出解決了
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
問題是被解決了,但是随之問題也就來了

同步函數用的是哪一個鎖呢?

函數需要被對象調用,那麼函數都有一個所屬對象的引用,就是this,是以同步函數所引用的鎖是this,我們來驗證一下,我們把程式改動一下 使用兩個線程來賣票,一個線程在同步代碼塊中,一個線程在同步函數中,都在執行賣票動作
當我們運作的時候就發現
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
他隻在show中進行,那是為什麼呢?因為主線程開啟的時候瞬間執行,我們要修改一下,讓線程1開啟的時候,主線程睡個10毫秒試試
這樣輸出的結果貌似是交替進行
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
但是所知而來的,是0票,這說明這個線程不安全,我們明明加了同步啊,怎麼還是不安全呢?因為他用的不是同一個鎖,一個用Object,一個是用this的鎖,我們再改動一下,我們把Object更好為this,這樣輸出
JAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是thisJAVA之旅(十三)——線程的安全性,synchronized關鍵字,多線程同步代碼塊,同步函數,同步函數的鎖是this
現在就安全,也正确了 好的,我們本篇幅就先到這裡了,我們下篇也繼續講線程