激情久久久_欧美视频区_成人av免费_不卡视频一二三区_欧美精品在欧美一区二区少妇_欧美一区二区三区的

腳本之家,腳本語言編程技術及教程分享平臺!
分類導航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服務器之家 - 腳本之家 - Golang - 詳解Golang五種原子性操作的用法

詳解Golang五種原子性操作的用法

2021-11-14 11:03kevinyan Golang

本文主要介紹了詳解Golang五種原子性操作的用法,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下

本文我們詳細聊一下Go語言的原子操作的用法,啥是原子操作呢?顧名思義,原子操作就是具備原子性的操作... 是不是感覺說了跟沒說一樣,原子性的解釋如下:

一個或者多個操作在 CPU 執行的過程中不被中斷的特性,稱為原子性(atomicity) 。這些操作對外表現成一個不可分割的整體,他們要么都執行,要么都不執行,外界不會看到他們只執行到一半的狀態。

CPU執行一系列操作時不可能不發生中斷,但如果我們在執行多個操作時,能讓他們的中間狀態對外不可見,那我們就可以宣稱他們擁有了"不可分割”的原子性。
類似的解釋我們在數據庫事務的ACID概念里也聽過,只不過這里保障原子性的執行體是CPU。

Go 語言提供了哪些原子操作

Go語言通過內置包sync/atomic提供了對原子操作的支持,其提供的原子操作有以下幾大類:

  • 增減,操作方法的命名方式為AddXXXType,保證對操作數進行原子的增減,支持的類型為int32、int64、uint32、uint64、uintptr,使用時以實際類型替換前面我說的XXXType就是對應的操作方法。
  • 載入,保證了讀取到操作數前沒有其他任務對它進行變更,操作方法的命名方式為LoadXXXType,支持的類型除了基礎類型外還支持Pointer,也就是支持載入任何類型的指針。
  • 存儲,有載入了就必然有存儲操作,這類操作的方法名以Store開頭,支持的類型跟載入操作支持的那些一樣。
  • 比較并交換,也就是CAS (Compare And Swap),像Go的很多并發原語實現就是依賴的CAS操作,同樣是支持上面列的那些類型。
  • 交換,這個簡單粗暴一些,不比較直接交換,這個操作很少會用。

互斥鎖跟原子操作的區別

平日里,在并發編程里,Go語言sync包里的同步原語Mutex是我們經常用來保證并發安全的,那么他跟atomic包里的這些操作有啥區別呢?在我看來他們在使用目的和底層實現上都不一樣:

  • 使用目的:互斥鎖是用來保護一段邏輯,原子操作用于對一個變量的更新保護。
  • 底層實現:Mutex由操作系統的調度器實現,而atomic包中的原子操作則由底層硬件指令直接提供支持,這些指令在執行的過程中是不允許中斷的,因此原子操作可以在lock-free的情況下保證并發安全,并且它的性能也能做到隨CPU個數的增多而線性擴展。

對于一個變量更新的保護,原子操作通常會更有效率,并且更能利用計算機多核的優勢。

比如下面這個,使用互斥鎖的并發計數器程序:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
func mutexAdd() {
 var a int32 =  0
 var wg sync.WaitGroup
 var mu sync.Mutex
 start := time.Now()
 for i := 0; i < 100000000; i++ {
  wg.Add(1)
  go func() {
   defer wg.Done()
   mu.Lock()
   a += 1
   mu.Unlock()
  }()
 }
 wg.Wait()
 timeSpends := time.Now().Sub(start).Nanoseconds()
 fmt.Printf("use mutex a is %d, spend time: %v\n", a, timeSpends)
}

把Mutex改成用方法atomic.AddInt32(&a, 1)調用,在不加鎖的情況下仍然能確保對變量遞增的并發安全。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
func AtomicAdd() {
 var a int32 =  0
 var wg sync.WaitGroup
 start := time.Now()
 for i := 0; i < 1000000; i++ {
  wg.Add(1)
  go func() {
   defer wg.Done()
   atomic.AddInt32(&a, 1)
  }()
 }
 wg.Wait()
 timeSpends := time.Now().Sub(start).Nanoseconds()
 fmt.Printf("use atomic a is %d, spend time: %v\n", atomic.LoadInt32(&a), timeSpends)
}

可以在本地運行以上這兩段代碼,可以觀察到計數器的結果都最后都是1000000,都是線程安全的。
需要注意的是,所有原子操作方法的被操作數形參必須是指針類型,通過指針變量可以獲取被操作數在內存中的地址,從而施加特殊的CPU指令,確保同一時間只有一個goroutine能夠進行操作。
上面的例子除了增加操作外我們還演示了載入操作,接下來我們來看一下CAS操作。

比較并交換

該操作簡稱CAS (Compare And Swap)。 這類操作的前綴為 CompareAndSwap :

?
1
2
3
func CompareAndSwapInt32(addr *int32, old, new int32) (swapped bool)
 
func CompareAndSwapPointer(addr *unsafe.Pointer, old, new unsafe.Pointer) (swapped bool)

該操作在進行交換前首先確保被操作數的值未被更改,即仍然保存著參數 old 所記錄的值,滿足此前提條件下才進行交換操作。CAS的做法類似操作數據庫時常見的樂觀鎖機制。

需要注意的是,當有大量的goroutine 對變量進行讀寫操作時,可能導致CAS操作無法成功,這時可以利用for循環多次嘗試。

上面我只列出了比較典型的int32和unsafe.Pointer類型的CAS方法,主要是想說除了讀數值類型進行比較交換,還支持對指針進行比較交換。

unsafe.Pointer提供了繞過Go語言指針類型限制的方法,unsafe指的并不是說不安全,而是說官方并不保證向后兼容。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
// 定義一個struct類型P
type P struct{ x, y, z int }
  
// 執行類型P的指針
var pP *P
  
func main() {
  
    // 定義一個執行unsafe.Pointer值的指針變量
    var unsafe1 = (*unsafe.Pointer)(unsafe.Pointer(&pP))
  
    // Old pointer
    var sy P
  
    // 為了演示效果先將unsafe1設置成Old Pointer
    px := atomic.SwapPointer(
        unsafe1, unsafe.Pointer(&sy))
  
    // 執行CAS操作,交換成功,結果返回true
    y := atomic.CompareAndSwapPointer(
        unsafe1, unsafe.Pointer(&sy), px)
  
    fmt.Println(y)
}

上面的示例并不是在并發環境下進行的CAS,只是為了演示效果,先把被操作數設置成了Old Pointer。
其實Mutex的底層實現也是依賴原子操作中的CAS實現的,原子操作的atomic包相當于是sync包里的那些同步原語的實現依賴。

比如互斥鎖Mutex的結構里有一個state字段,其是表示鎖狀態的狀態位。

?
1
2
3
4
type Mutex struct {
 state int32
 sema  uint32
}

為了方便理解,我們在這里將它的狀態定義為0和1,0代表目前該鎖空閑,1代表已被加鎖,以下是sync.Mutex中Lock方法的部分實現代碼。

?
1
2
3
4
5
6
7
8
9
10
11
func (m *Mutex) Lock() {
   // Fast path: grab unlocked mutex.
   if atomic.CompareAndSwapInt32(&m.state, 0, mutexLocked) {
       if race.Enabled {
           race.Acquire(unsafe.Pointer(m))
       }
       return
   }
   // Slow path (outlined so that the fast path can be inlined)
    m.lockSlow()
}

在atomic.CompareAndSwapInt32(&m.state, 0, mutexLocked)中,m.state代表鎖的狀態,通過CAS方法,判斷鎖此時的狀態是否空閑(m.state==0),是,則對其加鎖(mutexLocked常量的值為1)。

atomic.Value保證任意值的讀寫安全

atomic包里提供了一套Store開頭的方法,用來保證各種類型變量的并發寫安全,避免其他操作讀到了修改變量過程中的臟數據。

?
1
2
3
4
5
6
7
8
func StoreInt32(addr *int32, val int32)
 
func StoreInt64(addr *int64, val int64)
 
func StorePointer(addr *unsafe.Pointer, val unsafe.Pointer)
 
 
...

這些操作方法的定義與上面介紹的那些操作的方法類似,我就不再演示怎么使用這些方法了。

值得一提的是如果你想要并發安全的設置一個結構體的多個字段,除了把結構體轉換為指針,通過StorePointer設置外,還可以使用atomic包后來引入的atomic.Value,它在底層為我們完成了從具體指針類型到unsafe.Pointer之間的轉換。

有了atomic.Value后,它使得我們可以不依賴于不保證兼容性的unsafe.Pointer類型,同時又能將任意數據類型的讀寫操作封裝成原子性操作(中間狀態對外不可見)。

atomic.Value類型對外暴露了兩個方法:

  • v.Store(c) - 寫操作,將原始的變量c存放到一個atomic.Value類型的v里。
  • c := v.Load() - 讀操作,從線程安全的v中讀取上一步存放的內容。

1.17 版本我看還增加了Swap和CompareAndSwap方法。

簡潔的接口使得它的使用也很簡單,只需將需要做并發保護的變量讀取和賦值操作用Load()和Store()代替就行了。
由于Load()返回的是一個interface{}類型,所以在使用前我們記得要先轉換成具體類型的值,再使用。下面是一個簡單的

例子演示atomic.Value的用法。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
type Rectangle struct {
 length int
 width  int
}
 
var rect atomic.Value
 
func update(width, length int) {
 rectLocal := new(Rectangle)
 rectLocal.width = width
 rectLocal.length = length
 rect.Store(rectLocal)
}
 
func main() {
 wg := sync.WaitGroup{}
 wg.Add(10)
 // 10 個協程并發更新
 for i := 0; i < 10; i++ {
  go func() {
   defer wg.Done()
   update(i, i+5)
  }()
 }
 wg.Wait()
 _r := rect.Load().(*Rectangle)
 fmt.Printf("rect.width=%d\nrect.length=%d\n", _r.width, _r.length)
}

你也可以試試,不用atomic.Value,直接給Rectange類型的指針變量賦值,看看在并發條件下,兩個字段的值是不是能跟預期的一樣變成10和15。

總結

本文詳細介紹了Go語言原子操作atomic包中會被高頻使用的操作的使用場景和用法,當然我并沒有羅列atomic包里所有操作的用法,主要是考慮到有的用到的地方實在不多,或者是已經被更好的方式替代,還有就是覺得確實沒必要,看完本文的內容相信你已經完全具備自行探索atomic包的能力了。

再強調一遍,原子操作由底層硬件支持,而鎖則由操作系統的調度器實現。鎖應當用來保護一段邏輯,對于一個變量更新的保護,原子操作通常會更有效率,并且更能利用計算機多核的優勢,如果要更新的是一個復合對象,則應當使用atomic.Value封裝好的實現。

到此這篇關于詳解Golang五種原子性操作的用法的文章就介紹到這了,更多相關詳解Golang五種原子性操作的用法內容請搜索服務器之家以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持服務器之家! 

原文鏈接:https://juejin.cn/post/7010590496204521485

延伸 · 閱讀

精彩推薦
  • Golanggolang的httpserver優雅重啟方法詳解

    golang的httpserver優雅重啟方法詳解

    這篇文章主要給大家介紹了關于golang的httpserver優雅重啟的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,...

    helight2992020-05-14
  • GolangGolang通脈之數據類型詳情

    Golang通脈之數據類型詳情

    這篇文章主要介紹了Golang通脈之數據類型,在編程語言中標識符就是定義的具有某種意義的詞,比如變量名、常量名、函數名等等,Go語言中標識符允許由...

    4272021-11-24
  • GolangGolang中Bit數組的實現方式

    Golang中Bit數組的實現方式

    這篇文章主要介紹了Golang中Bit數組的實現方式,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    天易獨尊11682021-06-09
  • Golanggo日志系統logrus顯示文件和行號的操作

    go日志系統logrus顯示文件和行號的操作

    這篇文章主要介紹了go日志系統logrus顯示文件和行號的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    SmallQinYan12302021-02-02
  • Golanggolang json.Marshal 特殊html字符被轉義的解決方法

    golang json.Marshal 特殊html字符被轉義的解決方法

    今天小編就為大家分享一篇golang json.Marshal 特殊html字符被轉義的解決方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧 ...

    李浩的life12792020-05-27
  • Golanggolang如何使用struct的tag屬性的詳細介紹

    golang如何使用struct的tag屬性的詳細介紹

    這篇文章主要介紹了golang如何使用struct的tag屬性的詳細介紹,從例子說起,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看...

    Go語言中文網11352020-05-21
  • Golanggolang 通過ssh代理連接mysql的操作

    golang 通過ssh代理連接mysql的操作

    這篇文章主要介紹了golang 通過ssh代理連接mysql的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧...

    a165861639710342021-03-08
  • Golanggo語言制作端口掃描器

    go語言制作端口掃描器

    本文給大家分享的是使用go語言編寫的TCP端口掃描器,可以選擇IP范圍,掃描的端口,以及多線程,有需要的小伙伴可以參考下。 ...

    腳本之家3642020-04-25
主站蜘蛛池模板: 欧美国产第一页 | 久久精品片 | 精品中文字幕在线观看 | 久在线观看 | 久久精品日韩一区 | 亚洲网站在线 | 亚洲精品在线观看免费 | 亚洲婷婷日日综合婷婷噜噜噜 | 高清做爰免费无遮网站挡 | 在线观看中文字幕国产 | 欧美一区2区三区4区公司二百 | 久久久久se | 国产免费看片 | 精品国产一区二区亚洲人成毛片 | 日本成人午夜视频 | 国产一区二区欧美 | sm高h视频 | 最新影院 | 免费一级a毛片在线播放视 日日草夜夜操 | 亚州视频在线 | 精品亚洲视频在线 | 欧美日韩在线影院 | 日本在线高清 | 亚洲第一成人在线观看 | 一级毛片电影网 | 国产精品刺激对白麻豆99 | 九九热在线观看视频 | 99re热视频这里只精品 | 亚洲综合91 | 欧美成人免费看 | 激情小说色 | 国产精品久久久久久久久久三级 | 一日本道久久久精品国产 | 狠狠ri| 日本a∨精品中文字幕在线 被啪羞羞视频在线观看 | 日韩精品久 | 欧美日韩综合视频 | 久久精品欧美一区二区 | 91精品动漫在线观看 | 亚洲婷婷日日综合婷婷噜噜噜 | 国产在线免费 |