色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

您好,歡迎來電子發燒友網! ,新用戶?[免費注冊]

您的位置:電子發燒友網>源碼下載>數值算法/人工智能>

關于 Redis連接池解析

大小:0.4 MB 人氣: 2017-10-12 需要積分:1
 一、關于連接池
  一個數據庫服務器只擁有有限的資源,并且如果你沒有充分使用這些資源,你可以通過使用更多的連接來提高吞吐量。一旦所有的資源都在使用,那么你就不能通過增加更多的連接來提高吞吐量。事實上,吞吐量在連接負載較大時就開始下降了。通常可以通過限制與可用的資源相匹配的數據庫連接的數量來提高延遲和吞吐量。
  如果不使用連接池,那么,每次傳輸數據,我們都需要進行創建連接,收發數據,關閉連接。在并發量不高的場景,基本上不會有什么問題,一旦并發量上去了,那么,一般就會遇到下面幾個常見問題:
  性能普遍上不去CPU 大量資源被系統消耗網絡一旦抖動,會有大量 TIME_WAIT 產生,不得不定期重啟服務或定期重啟機器服務器工作不穩定,QPS 忽高忽低
  要想解決這些問題,我們就要用到連接池了。連接池的思路很簡單,在初始化時,創建一定數量的連接,先把所有長連接存起來,然后,誰需要使用,從這里取走,干完活立馬放回來。 如果請求數超出連接池容量,那么就排隊等待、退化成短連接或者直接丟棄掉。
  二、使用連接池遇到的坑
  最近在一個項目中,需要實現一個簡單的 Web Server 提供 Redis 的 HTTP interface,提供 JSON 形式的返回結果。考慮用 Go 來實現。
  首先,去看一下 Redis 官方推薦的 Go Redis driver。官方 Star 的項目有兩個:Radix.v2 和 Redigo。經過簡單的比較后,選擇了更加輕量級和實現更加優雅的 Radix.v2。
  Radix.v2 包是根據功能劃分成一個個的 sub package,每一個 sub package 在一個獨立的子目錄中,結構非常清晰。我的項目中會用到的 sub package 有 redis 和 pool。
  由于我想讓這種被 fork 的進程最好簡單點,做的事情單一一些,所以,在沒有深入去看 Radix.v2 的 pool 的實現之前,我選擇了自己實現一個 Redis pool。(這里,就不貼代碼了。后來發現自己實現的 Redis pool 與 Radix.v2 實現的 Redis pool 的原理是一樣的,都是基于 channel 實現的, 遇到的問題也是一樣的。)
  不過在測試過程中,發現了一個詭異的問題。在請求過程中經常會報 EOF 錯誤。而且是概率性出現,一會有問題,一會又好了。通過反復的測試,發現 bug 是有規律的,當程序空閑一會后,再進行連續請求,會發生3次失敗,然后之后的請求都能成功,而我的連接池大小設置的是3。再進一步分析,程序空閑300秒后,再請求就會失敗,發現我的 Redis server 配置了 timeout 300,至此,問題就清楚了。是連接超時 Redis server 主動斷開了連接。客戶端這邊從一個超時的連接請求就會得到 EOF 錯誤。
  然后我看了一下 Radix.v2 的 pool 包的源碼,發現這個庫本身并沒有檢測壞的連接,并替換為新的連接的機制。也就是說我每次從連接池里面 Get 的連接有可能是壞的連接。所以,我當時臨時的解決方案是通過增加失敗后自動重試來解決了。不過,這樣的處理方案,連接池的作用好像就沒有了。技術債能早點還的還是早點還上。
  三、使用連接池的正確姿勢
  想到我們的 ngx_lua 項目里面也大量使用 redis 連接池,他們怎么沒有遇到這個問題呢。只能去看看源碼了。
  經過抽象分離, ngx_lua 里面使用 redis 連接池部分的代碼大致是這樣的:
  server { location /pool { content_by_lua_block { localredis = require“resty.redis”localred = redis:new() localok, err = red:connect(“127.0.0.1”, 6379) ifnotok thenngx.say(“failed to connect: ”, err) returnendok, err = red:set(“hello”, “world”) ifnotok thenreturnendred:set_keepalive(10000, 100) } } }
  發現有個 set_keepalive 的方法,查了一下官方文檔,方法的原型是 syntax: ok, err = red:set_keepalive(max_idle_timeout, pool_size) 貌似 max_idle_timeout 這個參數,就是我們所缺少的東西,然后進一步跟蹤源碼,看看里面是怎么保證連接有效的。
  function_M.set_keepalive(self, 。..) local sock = self.sock ifnot sock then returnnil, “not initialized”end ifself.subscribed then returnnil, “subscribed state”end returnsock:setkeepalive(。..) end
  至此,已經清楚了,使用了 tcp 的 keepalive 心跳機制。
  于是,通過與 Radix.v2 的作者一些討論,選擇自己在 redis 這層使用心跳機制,來解決這個問題。
  四、最后的解決方案
  在創建連接池之后,起一個 goroutine,每隔一段 idleTime 發送一個 PING 到 Redis server。其中,idleTime 略小于 Redis server 的 timeout 配置。
  連接池初始化部分代碼如下:
  p, err:= pool.New(“tcp”, u.Host, concurrency) errHndlr(err) go func() { for{ p.Cmd(“PING”) time.Sleep(idelTime * time.Second) } }()
  使用 redis 傳輸數據部分代碼如下:
  func redisDo(p *pool.Pool, cmd string, args 。..interface{}) (reply *redis.Resp, errerror) { reply = p.Cmd(cmd, args.。.) iferr= reply.Err; err!= nil { iferr!= io.EOF { Fatal.Println(“redis”, cmd, args, “err is”, err) } } return }
  其中,Radix.v2 連接池內部進行了連接池內連接的獲取和放回,代碼如下:
  // Cmd automatically gets oneclient fromthepool, executes thegiven command// (returning its result), and puts the client back in the poolfunc (p *Pool) Cmd(cmd string, args 。..interface{}) *redis.Resp { c, err := p.Get() iferr != nil { returnredis.NewResp(err) } defer p.Put(c) returnc.Cmd(cmd, args.。.) }
  這樣,我們就有了 keepalive 的機制,不會出現 timeout 的連接了,從 redis 連接池里面取出的連接都是可用的連接了。看似簡單的代碼,卻完美的解決了連接池里面超時連接的問題。同時,就算 Redis server 重啟等情況,也能保證連接自動重連。
?

非常好我支持^.^

(1) 100%

不好我反對

(0) 0%

關于 Redis連接池解析下載

相關電子資料下載

      發表評論

      用戶評論
      評價:好評中評差評

      發表評論,獲取積分! 請遵守相關規定!

      ?
      主站蜘蛛池模板: 美女直播喷水| 久cao在线香蕉| 久久99精品国产麻豆婷婷| 三级网站视频在线观看| 7m凹凸国产刺激在线视频| 久久久精品免费免费直播| 亚洲视频在线观看不卡| 国产亚洲tv在线观看| 無码一区中文字幕少妇熟女网站| 亚洲电影第1页| 国产三级精品三级男人的天堂| 使劲别停好大好深好爽动态图| 妇少水多18P蜜泬17P亚洲乱| 日本高清无卡码一区二区久久| bl(高h)文| 青青伊人国产| 国产精品第一综合首页| 亚洲91av| 久久这里只有精品国产99| 18美女腿打开无遮软件| 欧美乱妇15p图| 国产精品97久久AV色婷婷| 亚洲精品九色在线网站| 精品国产午夜肉伦伦影院| 制服国产欧美亚洲日韩| 欧美18videosex性欧美老师| 成人午夜剧场| 亚洲 欧美 国产 视频二区| 久久99这里只有精品| 99精品电影| 水蜜桃亚洲一二三四在线| 好看AV中文字幕在线观看| 中文字幕精品AV内射夜夜夜| 啪啪啪社区| 国语大学生自产拍在线观看| 在线高清电影理论片4399| 欧美精品久久久久久久久大尺度| 大胸美女被cao哭| 亚洲午夜精品AV无码少妇| 免费久久狼人香蕉网| 国产互换后人妻的疯狂VIDEO|