Golang channel死鎖的幾種情況小結
死鎖是指兩個或兩個以上的協(xié)程的執(zhí)行過程中,由于競爭資源或由于彼此通信而造成的一種阻塞的現(xiàn)象,若無外力作用,他們將無法推進下去,以下是總結出來的幾種死鎖情況。
1.死鎖1:一個通道在一個主go程里同時進行讀和寫
2.死鎖2:go程開啟之前使用通道
3.死鎖3 :通道1中調用了通道2,通道2中調用通道1
4.死鎖4:直接讀取空channel的死鎖
5.死鎖5:超過channel緩存繼續(xù)寫入數(shù)據(jù)導致死鎖
6.向已關閉的channel中寫入數(shù)據(jù)不會導致死鎖,但是回出發(fā)panic異常
死鎖1:一個通道在一個主go程里同時進行讀和寫
func main() { // 死鎖1 ch := make(chan int) ch <- 100 num := <-ch fmt.Println("num=", num) }
還有另外一個場景:下面的場景是因為語句中的 <-ch1 是在主協(xié)程中先求值,會導致主協(xié)程阻塞。
package main import ( "fmt" "time" ) func main() { ch1 := make(chan int) go fmt.Println(<-ch1) ch1 <- 5 time.Sleep(1 * time.Second) }
死鎖2:go程開啟之前使用通道
func main() { ch := make(chan int) ch <- 100 //此處死鎖 優(yōu)于go程之前使用通道 go func() { num := <-ch fmt.Println("num=", num) }() //ch <- 100 此處不死鎖 time.Sleep(time.Second*3) fmt.Println("finish") }
死鎖3 :通道1中調用了通道2,通道2中調用通道1
// 死鎖3 func main() { ch1 := make(chan int ) ch2 := make(chan int ) go func() { for { select { case num := <-ch1: fmt.Println("num=", num) ch2 <- 100 } } }() for { select { case num := <-ch2: fmt.Println("num=", num) ch1 <- 300 } } }
死鎖4:讀取空channel 死鎖
func main() { // 死鎖1 ch := make(chan int) //close(ch) 向關閉的channel中讀取數(shù)據(jù) 是該數(shù)據(jù)的類型的零值 num := <-ch fmt.Println("num=", num) }
死鎖5:超過channel緩存繼續(xù)寫入數(shù)據(jù)導致死鎖
func main() { ch := make(chan int, 2) ch <- 1 ch <- 2 ch <- 3 num := <-ch fmt.Println("num=", num) }
死鎖6:向已關閉的channel中寫入數(shù)據(jù)不會導致死鎖,但是回出發(fā)panic異常
func main() { ch := make(chan int, 2) close(ch) ch <- 1 num := <-ch fmt.Println("num=", num) }
到此這篇關于Golang channel死鎖的幾種情況小結的文章就介紹到這了,更多相關Golang channel 死鎖內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
golang接口實現(xiàn)調用修改(值接收者指針接收者)場景詳解
這篇文章主要為大家介紹了golang接口實現(xiàn)調用修改值接收者指針接收者示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-08-08Go語言集成mysql驅動、調用數(shù)據(jù)庫、查詢數(shù)據(jù)操作示例
這篇文章主要介紹了Go語言集成mysql驅動、調用數(shù)據(jù)庫、查詢數(shù)據(jù)操作,結合實例形式分析了Go語言安裝mysql驅動包、連接mysql數(shù)據(jù)庫及查詢等相關操作技巧,需要的朋友可以參考下2019-06-06