欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

GO語言臨界資源安全問題的深入理解

 更新時間:2021年12月09日 10:24:08   作者:Davie老師  
臨界資源安全問題也就是以往我們常聽到的線程安全問題,本文詳細的介紹了GO語言臨界資源安全問題的深入理解,具有一定的參考價值,感興趣的小伙伴們可以參考一下

一、臨界資源

臨界資源: 指并發(fā)環(huán)境中多個進程/線程/協(xié)程共享的資源。

但是在并發(fā)編程中對臨界資源的處理不當, 往往會導致數(shù)據(jù)不一致的問題。

示例代碼:

package main
?
import (
    "fmt"
    "time"
)
?
func main()  {
    a := 1
    go func() {
        a = 2
        fmt.Println("子goroutine。。",a)
    }()
    a = 3
    time.Sleep(1)
    fmt.Println("main goroutine。。",a)
}

我們通過終端命令來執(zhí)行:

能夠發(fā)現(xiàn)一處被多個goroutine共享的數(shù)據(jù)。

二、臨界資源安全問題

并發(fā)本身并不復雜,但是因為有了資源競爭的問題,就使得我們開發(fā)出好的并發(fā)程序變得復雜起來,因為會引起很多莫名其妙的問題。

如果多個goroutine在訪問同一個數(shù)據(jù)資源的時候,其中一個線程修改了數(shù)據(jù),那么這個數(shù)值就被修改了,對于其他的goroutine來講,這個數(shù)值可能是不對的。

舉個例子,我們通過并發(fā)來實現(xiàn)火車站售票這個程序。一共有100張票,4個售票口同時出售。

我們先來看一下示例代碼:

package main
?
import (
    "fmt"
    "math/rand"
    "time"
)
?
//全局變量
var ticket = 10 // 100張票
?
func main() {
    /*
    4個goroutine,模擬4個售票口,4個子程序操作同一個共享數(shù)據(jù)。
     */
    go saleTickets("售票口1") // g1,100
    go saleTickets("售票口2") // g2,100
    go saleTickets("售票口3") //g3,100
    go saleTickets("售票口4") //g4,100
?
    time.Sleep(5*time.Second)
}
?
func saleTickets(name string) {
    rand.Seed(time.Now().UnixNano())
    //for i:=1;i<=100;i++{
    //  fmt.Println(name,"售出:",i)
    //}
    for { //ticket=1
        if ticket > 0 { //g1,g3,g2,g4
            //睡眠
            time.Sleep(time.Duration(rand.Intn(1000)) * time.Millisecond)
            // g1 ,g3, g2,g4
            fmt.Println(name, "售出:", ticket)  // 1 , 0, -1 , -2
            ticket--   //0 , -1 ,-2 , -3
        } else {
            fmt.Println(name,"售罄,沒有票了。。")
            break
        }
    }
}
?

我們?yōu)榱烁玫挠^察臨界資源問題,每個goroutine先睡眠一個隨機數(shù),然后再售票,我們發(fā)現(xiàn)程序的運行結果,還可以賣出編號為負數(shù)的票。

分析:

我們的賣票邏輯是先判斷票數(shù)的編號是否為負數(shù),如果大于0,然后我們就進行賣票,只不過在賣票錢先睡眠,然后再賣,假如說此時已經賣票到只剩最后1張了,某一個goroutine持有了CPU的時間片,那么它再片段是否有票的時候,條件是成立的,所以它可以賣票編號為1的最后一張票。但是因為它在賣之前,先睡眠了,那么其他的goroutine就會持有CPU的時間片,而此時這張票還沒有被賣出,那么第二個goroutine再判斷是否有票的時候,條件也是成立的,那么它可以賣出這張票,然而它也進入了睡眠。。其他的第三個第四個goroutine都是這樣的邏輯,當某個goroutine醒來的時候,不會再判斷是否有票,而是直接售出,這樣就賣出最后一張票了,然而其他的goroutine醒來的時候,就會陸續(xù)賣出了第0張,-1張,-2張。

這就是臨界資源的不安全問題。某一個goroutine在訪問某個數(shù)據(jù)資源的時候,按照數(shù)值,已經判斷好了條件,然后又被其他的goroutine搶占了資源,并修改了數(shù)值,等這個goroutine再繼續(xù)訪問這個數(shù)據(jù)的時候,數(shù)值已經不對了。

三、臨界資源安全問題的解決

要想解決臨界資源安全的問題,很多編程語言的解決方案都是同步。通過上鎖的方式,某一時間段,只能允許一個goroutine來訪問這個共享數(shù)據(jù),當前goroutine訪問完畢,解鎖后,其他的goroutine才能來訪問。

我們可以借助于sync包下的鎖操作。

示例代碼:

package main
?
import (
    "fmt"
    "math/rand"
    "time"
    "sync"
)
?
//全局變量
var ticket = 10 // 100張票
?
var wg sync.WaitGroup
var matex sync.Mutex // 創(chuàng)建鎖頭
?
func main() {
    /*
    4個goroutine,模擬4個售票口,4個子程序操作同一個共享數(shù)據(jù)。
     */
    wg.Add(4)
    go saleTickets("售票口1") // g1,100
    go saleTickets("售票口2") // g2,100
    go saleTickets("售票口3") //g3,100
    go saleTickets("售票口4") //g4,100
    wg.Wait()              // main要等待。。。
?
    //time.Sleep(5*time.Second)
}
?
func saleTickets(name string) {
    rand.Seed(time.Now().UnixNano())
    defer wg.Done()
    //for i:=1;i<=100;i++{
    //  fmt.Println(name,"售出:",i)
    //}
    for { //ticket=1
        matex.Lock()
        if ticket > 0 { //g1,g3,g2,g4
            //睡眠
            time.Sleep(time.Duration(rand.Intn(1000)) * time.Millisecond)
            // g1 ,g3, g2,g4
            fmt.Println(name, "售出:", ticket) // 1 , 0, -1 , -2
            ticket--                         //0 , -1 ,-2 , -3
        } else {
            matex.Unlock() //解鎖
            fmt.Println(name, "售罄,沒有票了。。")
            break
        }
        matex.Unlock() //解鎖
    }
}
?

運行結果:

四、寫在最后

在Go的并發(fā)編程中有一句很經典的話:不要以共享內存的方式去通信,而要以通信的方式去共享內存。

在Go語言中并不鼓勵用鎖保護共享狀態(tài)的方式在不同的Goroutine中分享信息(以共享內存的方式去通信)。而是鼓勵通過channel將共享狀態(tài)或共享狀態(tài)的變化在各個Goroutine之間傳遞(以通信的方式去共享內存),這樣同樣能像用鎖一樣保證在同一的時間只有一個Goroutine訪問共享狀態(tài)。

當然,在主流的編程語言中為了保證多線程之間共享數(shù)據(jù)安全性和一致性,都會提供一套基本的同步工具集,如鎖,條件變量,原子操作等等。Go語言標準庫也毫不意外的提供了這些同步機制,使用方式也和其他語言也差不多。

到此這篇關于GO語言臨界資源安全問題的深入理解的文章就介紹到這了,更多相關GO語言臨界資源安全 內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 在goland中讀取tpl文件的圖文操作

    在goland中讀取tpl文件的圖文操作

    這篇文章主要介紹了在goland中讀取tpl文件的圖文操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Go語言單線程運行也會有的并發(fā)問題解析

    Go語言單線程運行也會有的并發(fā)問題解析

    這篇文章主要為大家介紹了Go語言單線程運行的并發(fā)問題解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-12-12
  • Go語言使用組合的思想實現(xiàn)繼承

    Go語言使用組合的思想實現(xiàn)繼承

    這篇文章主要為大家詳細介紹了在 Go 里面如何使用組合的思想實現(xiàn)“繼承”,文中的示例代碼講解詳細,對我們學習Go語言有一定的幫助,需要的可以了解一下
    2022-12-12
  • 詳解Golang如何優(yōu)雅的終止一個服務

    詳解Golang如何優(yōu)雅的終止一個服務

    后端服務通常會需要創(chuàng)建子協(xié)程來進行相應的作業(yè),但進程接受到終止信號或正常結束時,并沒有判斷或等待子協(xié)程執(zhí)行結束,下面這篇文章主要給大家介紹了關于Golang如何優(yōu)雅的終止一個服務的相關資料,需要的朋友可以參考下
    2022-03-03
  • golang對自定義類型進行排序的解決方法

    golang對自定義類型進行排序的解決方法

    學習一門編程語言,要掌握原子數(shù)據(jù)類型,還需要掌握自定義數(shù)據(jù)類型。下面這篇文章主要給大家介紹了關于golang如何對自定義類型進行排序的相關資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考下。
    2017-12-12
  • 使用Go語言封裝實現(xiàn)郵件發(fā)送功能

    使用Go語言封裝實現(xiàn)郵件發(fā)送功能

    在現(xiàn)代 Web 開發(fā)中,郵件發(fā)送功能是一個常見的需求,本文將介紹如何在 Go 語言中封裝一個通用的郵件發(fā)送包,支持驗證碼發(fā)送和通用郵件發(fā)送,需要的可以參考下
    2025-03-03
  • 使用Go語言自制簡單易用的Web框架

    使用Go語言自制簡單易用的Web框架

    這篇文章主要為大家詳細介紹了如何使用Go語言實現(xiàn)自制簡單易用的Web框架,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2024-01-01
  • 詳解Go語言實現(xiàn)線性查找算法和二分查找算法

    詳解Go語言實現(xiàn)線性查找算法和二分查找算法

    線性查找又稱順序查找,它是查找算法中最簡單的一種。二分查找,也稱折半查找,相比于線性查找,它是一種效率較高的算法。本文將用Go語言實現(xiàn)這兩個查找算法,需要的可以了解一下
    2022-12-12
  • go語言題解LeetCode1299將每個元素替換為右側最大元素

    go語言題解LeetCode1299將每個元素替換為右側最大元素

    這篇文章主要為大家介紹了go語言LeetCode刷題1299將每個元素替換為右側最大元素示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-01-01
  • 簡單講解Go程序中使用MySQL的方法

    簡單講解Go程序中使用MySQL的方法

    這篇文章主要介紹了Go程序中使用MySQL的方法,需要使用第三方包來進行連接,需要的朋友可以參考下
    2015-10-10

最新評論