Redis鎖構造

来源:https://www.cnblogs.com/linxiyue/archive/2018/01/04/8185933.html
-Advertisement-
Play Games

單線程與隔離性 Redis是使用單線程的方式來執行事務的,事務以串列的方式運行,也就是說Redis中單個命令的執行和事務的執行都是線程安全的,不會相互影響,具有隔離性。 在多線程編程中,對於共用資源的訪問要十分的小心: 在不加鎖的情況下,num是不能保持為1的。 而在Redis中,併發執行單個命令具 ...


單線程與隔離性

Redis是使用單線程的方式來執行事務的,事務以串列的方式運行,也就是說Redis中單個命令的執行和事務的執行都是線程安全的,不會相互影響,具有隔離性。

在多線程編程中,對於共用資源的訪問要十分的小心:

import threading

num = 1
lock = threading.Lock()


def change_num():
    global num
    for i in xrange(100000):
        #lock.acquire()
        num += 5
        num -= 5
        #lock.release()


if __name__ == '__main__':
    pool = [threading.Thread(target=change_num) for i in xrange(5)]
    for t in pool:
        t.start()
    for t in pool:
        t.join()
    print num

在不加鎖的情況下,num是不能保持為1的。

而在Redis中,併發執行單個命令具有很好的隔離性:

import redis

conn = redis.StrictRedis(host="localhost", port=6379, db=1)
conn.set('num', 1)


def change_num(conn):
    for i in xrange(100000):
    ┆   conn.incr('num', 5)
    ┆   conn.decr('num', 5)


if __name__ == '__main__':
    conn_pool = [redis.StrictRedis(host="localhost", port=6379, db=1)
                 for i in xrange(5)]
    t_pool = []
    for conn in conn_pool:
        t = threading.Thread(target=change_num, args=(conn,))
        t_pool.append(t)
    for t in t_pool:
        t.start()
    for t in t_pool:
        t.join()
    print conn.get('num')

模擬的5個客戶端同時對Redis中的num值進行操作,num最終結果會保持為1:

1
real	0m46.463s
user	0m28.748s
sys	0m6.276s

利用Redis中單個操作和事務的原子性可以做很多事情,最簡單的就是做全局計數器了。

比如在簡訊驗證碼業務中,要限制一個用戶在一分鐘內只能發送一次,如果使用關係型資料庫,需要為每個手機號記錄上次發送簡訊的時間,當用戶請求驗證碼時,取出與當前時間進行對比。

這一情況下,當用戶短時間點擊多次時,不僅增加了資料庫壓力,而且還會出現同時查詢均符合條件但資料庫更新簡訊發送時間較慢的問題,就會重覆發送簡訊了。

在Redis中解決這一問題就很簡單,只需要用手機號作為key創建一個生存期限為一分鐘的數值即可。key不存在時能發送簡訊,存在時則不能發送簡訊:

def can_send(phone):
    key = "message:" + str(phone)
    if conn.set(key, 0, nx=True, ex=60):
    ┆   return True
    else:
    ┆   return False

至於一些不可名的30分鐘內限制訪問或者下載5次的功能,將用戶ip作為key,值設為次數上限,過期時間設為限制時間,每次用戶訪問時自減即可:

def can_download(ip):
    key = "ip:" + str(ip)
    conn.set(key, 5, nx=True, ex=600)
    if conn.decr(key) >= 0:
    ┆   return True
    else:
    ┆   return False

Redis基本事務與樂觀鎖

雖然Redis單個命令具有原子性,但當多個命令並行執行的時候,會有更多的問題。

比如舉一個轉賬的例子,將用戶A的錢轉給用戶B,那麼用戶A的賬戶減少需要與B賬戶的增多同時進行:

import threading
import time

import redis

conn = redis.StrictRedis(host="localhost", port=6379, db=1)
conn.mset(a_num=10, b_num=10)


def a_to_b():
    if int(conn.get('a_num')) >= 10:
        conn.decr('a_num', 10)
        time.sleep(.1)
        conn.incr('b_num', 10)
    print conn.mget('a_num', "b_num")


def b_to_a():
    if int(conn.get('b_num')) >= 10:
        conn.decr('b_num', 10)
        time.sleep(.1)
        conn.incr('a_num', 10)
    print conn.mget('a_num', "b_num")


if __name__ == '__main__':
    pool = [threading.Thread(target=a_to_b) for i in xrange(3)]
    for t in pool:
        t.start()

    pool = [threading.Thread(target=b_to_a) for i in xrange(3)]
    for t in pool:
        t.start()

運行結果:

['0', '10']
['0', '10']
['0', '0']
['0', '0']
['0', '10']
['10', '10']

出現了賬戶總額變少的情況。雖然是人為的為自增自減命令之間添加了100ms延遲,但在實際併發很高的情況中是很可能出現的,兩個命令執行期間執行了其它的語句。

那麼現在要保證的是兩個增減命令執行期間不受其它命令的干擾,Redis的事務可以達到這一目的。

Redis中,被MULTI命令和EXEC命令包圍的所有命令會一個接一個的執行,直到所有命令都執行完畢為止。一個事務完畢後,Redis才會去處理其它的命令。也就是說,Redis事務是具有原子性的。

python中可以用pipeline來創建事務:

def a_to_b():
    if int(conn.get('a_num')) >= 10:
    ┆   pipeline = conn.pipeline()
    ┆   pipeline.decr('a_num', 10)
    ┆   time.sleep(.1)
    ┆   pipeline.incr('b_num', 10)
    ┆   pipeline.execute()
    print conn.mget('a_num', "b_num")


def b_to_a():
    if int(conn.get('b_num')) >= 10:
    ┆   pipeline = conn.pipeline()
    ┆   pipeline.decr('b_num', 10)
    ┆   time.sleep(.1)
    ┆   pipeline.incr('a_num', 10)
    ┆   pipeline.execute()
    print conn.mget('a_num', "b_num")

結果:

['0', '20']
['10', '10']
 ['-10', '30']
['-10', '30']
['0', '20']
['10', '10']

可以看到,兩條語句確實一起執行了,賬戶總額不會變,但出現了負值的情況。這是因為事務在exec命令被調用之前是不會執行的,所以用讀取的數據做判斷與事務執行之間就有了時間差,期間實際數據發生了變化。

為了保持數據的一致性,我們還需要用到一個事務命令WATCH。WATCH可以對一個鍵進行監視,監視後到EXEC命令執行之前,如果被監視的鍵值發生了變化(替換,更新,刪除等),EXEC命令會返回一個錯誤,而不會真正的執行:

>>> pipeline.watch('a_num')
True
>>> pipeline.multi()
>>> pipeline.incr('a_num',10)
StrictPipeline<ConnectionPool<Connection<host=localhost,port=6379,db=1>>>
>>> pipeline.execute()
[20]
>>> pipeline.watch('a_num')
True
>>> pipeline.incr('a_num',10) #監視期間改變被監視鍵的值
30
>>> pipeline.multi()
>>> pipeline.incr('a_num',10)
StrictPipeline<ConnectionPool<Connection<host=localhost,port=6379,db=1>>>
>>> pipeline.execute()
    raise WatchError("Watched variable changed.")
redis.exceptions.WatchError: Watched variable changed.

現在為代碼加上watch:

def a_to_b():
      pipeline = conn.pipeline()
      try:
      ┆   pipeline.watch('a_num')
      ┆   if int(pipeline.get('a_num')) < 10:
      ┆   ┆   pipeline.unwatch()
      ┆   ┆   return
      ┆   pipeline.multi()
      ┆   pipeline.decr('a_num', 10)
      ┆   pipeline.incr('b_num', 10)
      ┆   pipeline.execute()
      except redis.exceptions.WatchError:
      ┆   pass
      print conn.mget('a_num', "b_num")
  
  
  def b_to_a():
      pipeline = conn.pipeline()
      try:
      ┆   pipeline.watch('b_num')
      ┆   if int(pipeline.get('b_num')) < 10:
      ┆   ┆   pipeline.unwatch()
      ┆   ┆   return
      ┆   pipeline.multi()
      ┆   pipeline.decr('b_num', 10)
      ┆   pipeline.incr('a_num', 10)
      ┆   pipeline.execute()
      except redis.exceptions.WatchError:
      ┆   pass
      print conn.mget('a_num', "b_num")

結果:

['0', '20']
['10', '10']
['20', '0']

成功實現了賬戶轉移,但是有三次嘗試失敗了,如果要儘可能的使每次交易都獲得成功,可以加嘗試次數或者嘗試時間:

def a_to_b():
    pipeline = conn.pipeline()
    end = time.time() + 5
    while time.time() < end:
    ┆   try:
    ┆   ┆   pipeline.watch('a_num')
    ┆   ┆   if int(pipeline.get('a_num')) < 10:
    ┆   ┆   ┆   pipeline.unwatch()
    ┆   ┆   ┆   return
    ┆   ┆   pipeline.multi()
    ┆   ┆   pipeline.decr('a_num', 10)
    ┆   ┆   pipeline.incr('b_num', 10)
    ┆   ┆   pipeline.execute()
    ┆   ┆   return True
    ┆   except redis.exceptions.WatchError:
    ┆   ┆   pass
    return False

這樣,Redis可以使用事務實現類似於鎖的機制,但這個機制與關係型資料庫的鎖有所不同。關係型資料庫對被訪問的數據行進行加鎖時,其它客戶端嘗試對被加鎖數據行進行寫入是會被阻塞的。

Redis執行WATCH時並不會對數據進行加鎖,如果發現數據已經被其他客戶端搶先修改,只會通知執行WATCH命令的客戶端,並不會阻止修改,這稱之為樂觀鎖。

用SET()構建鎖

用WACTH實現的樂觀鎖一般情況下是適用的,但存在一個問題,程式會為完成一個執行失敗的事務而不斷地進行重試。當負載增加的時候,重試次數會上升到一個不可接受的地步。

如果要自己正確的實現鎖的話,要避免下麵幾個情況:

  • 多個進程同時獲得了鎖
  • 持有鎖的進程在釋放鎖之前崩潰了,而其他進程卻不知道
  • 持有鎖的進行運行時間過長,鎖被自動釋放了,進程本身不知道,還會嘗試去釋放鎖

Redis中要實現鎖,需要用到一個命令,SET()或者說是SETNX()。SETNX只會在鍵不存在的情況下為鍵設置值,現在SET命令在加了NX選項的情況下也能實現這個功能,而且還能設置過期時間,簡直就是天生用來構建鎖的。

只要以需要加鎖的資源名為key設置一個值,要獲取鎖時,檢查這個key存不存在即可。若存在,則資源已被其它進程獲取,需要阻塞到其它進程釋放,若不存在,則建立key並獲取鎖:

import time
import uuid


class RedisLock(object):

    def __init__(self, conn, lockname, retry_count=3, timeout=10,):
        self.conn = conn
        self.lockname = 'lock:' + lockname
        self.retry_count = int(retry_count)
        self.timeout = int(timeout)
        self.unique_id = str(uuid.uuid4())

    def acquire(self):
        retry = 0
        while retry < self.retry_count:
            if self.conn.set(lockname, self.unique_id, nx=True, ex=self.timeout):
                return self.unique_id
            retry += 1
            time.sleep(.001)
        return False

    def release(self):
        if self.conn.get(self.lockname) == self.unique_id:
            self.conn.delete(self.lockname)
            return True
        else:
            return False

獲取鎖的預設嘗試次數限制3次,3次獲取失敗則返回。鎖的生存期限預設設為了10s,若不主動釋放鎖,10s後鎖會自動消除。

還保存了獲取鎖時鎖設置的值,當釋放鎖的時候,會先判斷保存的值和當前鎖的值是否一樣,如果不一樣,說明是鎖過期被自動釋放然後被其它進程獲取了。所以鎖的值必須保持唯一,以免釋放了其它程式獲取的鎖。

使用鎖:

def a_to_b():
    lock = Redlock(conn, 'a_num')
    if not lock.acquire():
    ┆   return False

    pipeline = conn.pipeline()
    try:
    ┆   pipeline.get('a_num')
    ┆   (a_num,) = pipeline.execute()
    ┆   if int(a_num) < 10: 
    ┆   ┆   return False
    ┆   pipeline.decr('a_num', 10) 
    ┆   pipeline.incr('b_num', 10) 
    ┆   pipeline.execute()
    ┆   return True
    finally:
    ┆   lock.release()

釋放鎖時也可以用Lua腳本來告訴Redis:刪除這個key當且僅當這個key存在而且值是我期望的那個值:

    unlock_script = """
    if redis.call("get",KEYS[1]) == ARGV[1] then
    ┆   return redis.call("del",KEYS[1])
    else
    ┆   return 0
    end"""

可以用conn.eval來運行Lua腳本:

    def release(self):
    ┆   self.conn.eval(unlock_script, 1, self.lockname, self.unique_id)

這樣,一個Redis單機鎖就實現了。我們可以用這個鎖來代替WATCH,或者與WACTH同時使用。

實際使用中還要根據業務來決定鎖的粒度的問題,是鎖住整個結構還是鎖住結構中的一小部分。

粒度越大,性能越差,粒度越小,發生死鎖的幾率越大。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 架構師寫的軟體指南 《 程式員必讀之軟體架構 》筆記 語境 意圖 這個軟體項目/產品/系統是關於什麼的? 構建的是什麼? 它如何融入現有環境? 誰在使用? 功能性概覽 意圖 系統實際上做什麼? 哪些特性、功能、用例、用戶故事是重要的?原因? 重要用戶是誰?系統如何滿足他們的需求? 上述已經用於塑造和 ...
  • 高可用的兩大目的:數據備份,數據分片 1、FastDFS安裝配置 先配置一臺,將其中的配置文件打包,下載,然後配置其他機器時只需要解壓即可, 打包命令 然後下載,上傳到其他機器相對應的/etc目錄下 將其他機器中的fdfs文件夾刪除 解壓上傳文件 2、 伺服器列表 伺服器IP 組 角色 192.16 ...
  • 預覽數據 這次我們使用 Artworks.csv ,我們選取 100 行數據來完成本次內容。具體步驟: DataFrame 是 Pandas 內置的數據展示的結構,展示速度很快,通過 DataFrame 我們就可以快速的預覽和分析數據。代碼如下: 統計日期數據 我們仔細觀察一下 Date 列的數據, ...
  • 原文地址: 本文地址:http://www.cnblogs.com/aiweixiao/p/8202360.html Original 2018-01-02 關註 微信公眾號 程式員的文娛情懷 1.概述 常見的排序演算法,雖然很基礎,但是很見功力,如果能思路清晰,很快寫出來各個演算法的代碼實現,還是需要 ...
  • 本文目錄:1.幾個基本的概念2.創建線程的兩種方法3.線程相關的常用方法4.多線程安全問題和線程同步 4.1 多線程安全問題 4.2 線程同步 4.3 同步代碼塊和同步函數的區別以及鎖是什麼 4.4 單例懶漢模式的多線程安全問題5.死鎖(DeadLock) 1.幾個基本的概念 本文涉及到的一些概念, ...
  • 第一步:我們先來安裝Python 下載地址是:https://www.python.org/downloads/ 第二步,添加運行環境 手動添加環境變數:滑鼠右鍵我的電腦 -> 屬性 -> 點擊高級系統設置 -> 點擊環境變數 -> 點擊PATH 第三步:我們安裝pip 下載地址是:https:// ...
  • C++11 引入了 auto 和 decltype 這兩個關鍵字實現了類型推導,讓編譯器來操心變數的類型。這使得 C++ 也具有了和其他現代編程語言一樣,某種意義上提供了無需操心變數類型的使用習慣。 一. auto C++11之前:如果一個變數沒有聲明為 register變數,將自動被視為一個 au... ...
  • 位運算針對的是二進位,所以需要將進行位運算的數現轉成在記憶體中二進位的表示形式 左移或右移 例如: 3 << 2 = 12 原理就是: 左移就是從左邊開始去掉幾位,就在最後面添加0,補成32位 右移同理,在前面補0還是1要看最高位(最左邊)是0還是1。 計算方法: 左移:往左移幾位就乘以2的幾次冪 ( ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...