国产片侵犯亲女视频播放_亚洲精品二区_在线免费国产视频_欧美精品一区二区三区在线_少妇久久久_在线观看av不卡

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

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

服務器之家 - 腳本之家 - Golang - golang模擬實現帶超時的信號量示例代碼

golang模擬實現帶超時的信號量示例代碼

2020-05-08 11:14魚兒 Golang

這篇文章主要給大家介紹了關于golang模擬實現帶超時的信號量的相關資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面跟著小編來一起學習學習吧。

前言

最近在寫項目,需要用到信號量等待一些資源完成,但是最多等待N毫秒。在看本文的正文之前,我們先來看下C語言里的實現方法。

在C語言里,有如下的API來實現帶超時的信號量等待:

?
1
2
3
4
5
SYNOPSIS
  #include <pthread.h>
 
  int
  pthread_cond_timedwait(pthread_cond_t *cond, pthread_mutex_t *mutex, const struct timespec *abstime);

然后在查看golang的document后,發現golang里并沒有實現帶超時的信號量,官方文檔在這里

原理

我的業務場景是這樣的:我有一個緩存字典,當多個用戶請求1個不存在的key時,只有1個請求會穿透到后端,而所有用戶都要排隊等這個請求完成,或者超時返回。

怎么實現呢?其實稍微想一想cond的原理,就能模擬一個帶超時的cond出來。

在golang里,要同時實現”掛起等待”和”超時返回”,一般得用select case語法,一個case等待阻塞的資源,一個case等待一個timer,這一點是非常確定的。

原本阻塞的資源應該通過條件變量的機制來實現完成通知,既然這里決定用select case,那么自然想到用channel來代替這個完成通知。

接下來的問題就是,很多請求者并發來獲取這個資源,但是資源還沒有準備好,所以大家都要排隊并掛起,等待資源完成,并且當資源完成后通知大家。

所以,這里很自然要為這個資源做一個隊列,每個請求者創建一個chan,并將chan放到隊列里,接著select case等待這個chan的通知。而另一端,資源完成后遍歷隊列,通知每個chan即可。

最后一個問題是,只有第一個請求者才能穿透請求到后端,而后續請求者不應該穿透重復的請求,這可以通過判斷緩存里是否有這個key作為判定首次的條件,而標記位init來判斷請求者是否應該排隊。

我的場景

上面是思路,下面是我的業務場景實現。

?
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
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
func (cache *Cache) Get(key string, keyType int) *string {
 if keyType == KEY_TYPE_DOMAIN {
 key = "#" + key
 } else {
 key = "=" + key
 }
 
 cache.mutex.Lock()
 item, existed := cache.dict[key]
 if !existed {
 item = &cacheItem{}
 item.key = &key
 item.waitQueue = list.New()
 cache.dict[key] = item
 }
 cache.mutex.Unlock()
 
 conf := config.GetConfig()
 
 lastGet := getCurMs()
 
 item.mutex.Lock()
 item.lastGet = lastGet
 if item.init { // 已存在并且初始化
 defer item.mutex.Unlock()
 return item.value
 }
 
 // 未初始化,排隊等待結果
 wait := waitItem{}
 wait.wait_chan = make(chan *string, 1)
 item.waitQueue.PushBack(&wait)
 item.mutex.Unlock()
 
 // 新增key, 啟動goroutine獲取初始值
 if !existed {
 go cache.initCacheItem(item, keyType)
 }
 
 timer := time.NewTimer(time.Duration(conf.Cache_waitTime) * time.Millisecond)
 
 var retval *string = nil
 
 // 等待初始化完成
 select {
 case retval = <- wait.wait_chan:
 case <- timer.C:
 }
 return retval
}

簡述一下整個過程:

  • 首先鎖字典,如果key不存在,說明我是第一個請求者,我會創建這個key對應的value,只不過init=false表示它正在初始化。最后,釋放字典鎖。
  • 接下來,鎖住這個key,判斷它已經初始化完成,那么直接返回value。否則,創建一個chan放入waitQueue等待隊列。最后,釋放key鎖。
  • 接著,如果當前是第一個請求者,那么會穿透請求到后端(在一個獨立的協程里去發起網絡調用)。
  • 現在,創建一個用于超時的定時器。
  • 最后,無論當前是否是key的第一個請求者,還是初始化期間的并發請求者,它們都通過select case超時的等待結果完成。

在initCacheItem函數里,數據已獲取成功

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
// 一旦標記為init, 后續請求將不再操作waitQueue
item.mutex.Lock()
item.value = newValue
item.init = true
item.expire = expire
item.mutex.Unlock()
 
// 喚醒所有排隊者
waitQueue := item.waitQueue
for elem := waitQueue.Front(); elem != nil; elem = waitQueue.Front() {
wait := elem.Value.(*waitItem)
wait.wait_chan <- newValue
waitQueue.Remove(elem)
}
  • 首先,鎖住key,標記init=true,并賦值value,并釋放鎖。此后的請求,都可以立即返回,無需排隊。
  • 之后,因為init=true已被標記,此刻再也有沒有請求會修改waitQueue,所以無需加鎖,直接遍歷隊列,通知其中的每個chan。

最后

這樣就實現了帶超時的條件變量效果,實際上我的場景是一個broadcast的cond例子,大家可以參照思路實現自己想要的效果,活學活用。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。

原文鏈接:https://yuerblog.cc/2017/09/04/golang-condition-variable-with-timeout/

延伸 · 閱讀

精彩推薦
  • GolangGolang中Bit數組的實現方式

    Golang中Bit數組的實現方式

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

    天易獨尊11682021-06-09
  • Golanggolang的httpserver優雅重啟方法詳解

    golang的httpserver優雅重啟方法詳解

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

    helight2992020-05-14
  • Golanggo語言制作端口掃描器

    go語言制作端口掃描器

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

    腳本之家3642020-04-25
  • Golanggolang 通過ssh代理連接mysql的操作

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

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

    a165861639710342021-03-08
  • Golanggolang如何使用struct的tag屬性的詳細介紹

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

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

    Go語言中文網11352020-05-21
  • Golanggolang json.Marshal 特殊html字符被轉義的解決方法

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

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

    李浩的life12792020-05-27
  • Golanggo日志系統logrus顯示文件和行號的操作

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

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

    SmallQinYan12302021-02-02
  • GolangGolang通脈之數據類型詳情

    Golang通脈之數據類型詳情

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

    4272021-11-24
Weibo Article 1 Weibo Article 2 Weibo Article 3 Weibo Article 4 Weibo Article 5 Weibo Article 6 Weibo Article 7 Weibo Article 8 Weibo Article 9 Weibo Article 10 Weibo Article 11 Weibo Article 12 Weibo Article 13 Weibo Article 14 Weibo Article 15 Weibo Article 16 Weibo Article 17 Weibo Article 18 Weibo Article 19 Weibo Article 20 Weibo Article 21 Weibo Article 22 Weibo Article 23 Weibo Article 24 Weibo Article 25 Weibo Article 26 Weibo Article 27 Weibo Article 28 Weibo Article 29 Weibo Article 30 Weibo Article 31 Weibo Article 32 Weibo Article 33 Weibo Article 34 Weibo Article 35 Weibo Article 36 Weibo Article 37 Weibo Article 38 Weibo Article 39 Weibo Article 40
主站蜘蛛池模板: 亚洲精品9999 | 亚洲激情在线 | a级毛片免费在线 | 26uuu国产电影一区二区 | 亚洲成av人影片在线观看 | 国产精品一码二码三码在线 | 在线高清av | 在线欧美一区 | 精品久久久久久久久久久久 | 亚洲社区在线 | 成人免费大片黄在线播放 | 久久综合久久综合久久综合 | 日韩欧美国产一区二区三区 | 国产午夜精品一区二区三区嫩草 | 久久精品高清 | 欧美福利在线 | 久久大陆 | 亚洲精品国产乱码在线看蜜月 | 国产电影一区二区三区 | 亚洲成人久久久 | 精品国产91亚洲一区二区三区www | 国产一区免费 | 成人毛片在线观看视频 | 久久久久国产精品www | 国产成人精品视频 | 国产精品毛片一区二区三区 | 国产毛片视频 | 精品亚洲永久免费精品 | 精品伦精品一区二区三区视频 | 国产成人黄色 | 成人精品视频免费 | 成人a级片在线观看 | 亚洲 中文 欧美 日韩 在线观看 | 国产亚洲一区二区三区 | 污污视频网站免费 | 无码一区二区三区视频 | 久久99视频这里只有精品 | 国产在线拍揄自揄拍视频 | 成人在线观看网 | 亚洲生活片 | 久久99这里只有精品 |