


Ein Artikel, der ausführlich erklärt, wie man Redis zum Implementieren verteilter Sperren verwendet
Empfohlenes Lernen: Redis-Video-Tutorial
1. Was ist eine verteilte Sperre? Wenn wir Multithread-Code schreiben, konkurrieren möglicherweise verschiedene Threads um Ressourcen Sperren Sie die Ressource, und nur der Thread, der die Sperre erhalten hat, kann mit der Ausführung fortfahren.
Die Sperre im Prozess ist im Wesentlichen eine Variable im Speicher. Wenn ein Thread eine Operation zur Beantragung einer Sperre ausführt und der Wert der Variablen, die die Sperre darstellt, erfolgreich auf 1 gesetzt werden kann, bedeutet dies, dass die Sperre erfolgt ist erhalten, und andere Threads möchten es erhalten, wenn es gesperrt wird, und nachdem der Thread, der die Sperre besitzt, den Vorgang abschließt, setzt er den Wert der Sperre auf 0, was bedeutet, dass die Sperre aufgehoben wird.
Wovon wir oben sprechen, ist die Sperre zwischen verschiedenen Threads im Prozess eines Servers. Diese Sperre wird im Speicher platziert, und bei verteilten Anwendungen werden verschiedene Anwendungen (Prozesse oder Threads) auf verschiedenen Servern bereitgestellt Sperren können nicht durch Variablen im Speicher dargestellt werden.
Da die Sperre nun durch den gemeinsam genutzten Speicherbereich auf einem Server dargestellt werden kann, kann für verteilte Anwendungen das Speichersystem gemeinsam genutzt werden, um eine gemeinsam genutzte Sperre und Redis
zu speichern Als In-Memory-Datenbank wird sie sehr schnell ausgeführt und eignet sich sehr gut als gemeinsames Speichersystem zur Implementierung verteilter Sperren.
Redis
作为内存数据库,执行非常快,很适合作为实现分布式锁的共享存储系统。
2. 使用Redis实现分布式锁
对于一个锁来说,其实只有两个操作,加锁和释放锁,下面我们看来看通过Redis
要怎么实现?
2.1 加锁
Redis
的setnx
命令会判断键值是否存在,如果存在则不做任何操作,并返回0,如果不存在,则创建并赋值,并返回1,因此我们可以执行setnx
为一个代表锁键设置值,如果能设置成功,则表示获得锁,失败则无法获得锁。
# 使用key为lock来表示一个锁 setnx lock 1
2.2 释放锁
当执行好操作之后,要释放锁的时候直接把Redis
里的键值lock
删除就可以了,这样其他进程才能通过setnx
命令重新设置并获得该锁。
# 释放锁 del lock
通过上面两个命令,我们实现了一个简单的分布式锁,但这里就出现了一个问题:如果一个进程通过setnx
命令加锁之后,在执行具体操作出错了,没有办法及时释放锁,那么其他进程就无法获得该锁,系统便无法继续往下执行,解决这个问题的办法就是为锁设置一个有效期,在这个有效期之后,自动释放锁。
2.3 给锁设置有效期
给锁设置有效期非常简单,直接使用Redis
的expire
命令就可以了,如:
# 加锁 setnx lock 1 # 给锁设置10s有效期 expire lock 10
但是,现在又出现另一个问题了,如果我们在设置了锁之后,执行expire
命令之前该进程挂掉了,那么expire
就没有执行成功,锁一样是没有被释放掉的,所以一定要保证上面两个命令要一起执行,怎么保证呢?
有两个方法,一个是使用LUA
语言编写的脚本,另一个是使用Redis
的set
命令,set
命令后面跟nx
参数后,执行的效果与setnx
一致,且set
命令可以跟ex
参数来设置过期时间,所以我们可以使用set
命令把setnx
和expire
两个合并在一起,这样就可以保证执行的原子性了。
# 判断是否键值是否存在,ex后面跟着的是键值的有效期,10s set lock 1 nx ex 10
解决了锁的有效问题,现在我们再来看另外一个问题。
如上图所示,现在有A
,B
,C
Redis
implementiert. 🎜2.1 Sperren
🎜Redis
s Befehl setnx
ermittelt, ob der Schlüsselwert vorhanden ist, nichts wird ausgeführt und gibt 0 zurück. Wenn er nicht existiert, erstellen und weisen Sie einen Wert zu und geben Sie 1 zurück, damit wir setnx
ausführen können, um einen Wert für einen repräsentativen Sperrschlüssel festzulegen Wenn die Einstellung erfolgreich ist, bedeutet dies, dass die Sperre erhalten wurde. Fehler Die Sperre kann nicht erhalten werden. 🎜# rand_uid表示唯一id set lock rand_id nx ex 10
2.2 Sperre aufheben
🎜Wenn Sie nach dem Ausführen des Vorgangs die Sperre aufheben möchten, ändern Sie direkt den Schlüsselwertlock< in <code>Redis< /code> /code>Löschen Sie es einfach, damit andere Prozesse die Sperre über den Befehl <code>setnx
zurücksetzen und erhalten können. 🎜if redis.call("get",KEYS[1]) == ARGV[1] then return redis.call("del",KEYS[1]) else return 0 end
setnx
gesperrt wird, tritt während des spezifischen Vorgangs ein Fehler auf. Wenn es keine Möglichkeit gibt, die Sperre rechtzeitig aufzuheben, können andere Prozesse die Sperre nicht erhalten und das System kann die Ausführung nicht fortsetzen. Die Lösung dieses Problems besteht darin, einen Gültigkeitszeitraum für die Sperre festzulegen. und nach Ablauf dieser Gültigkeitsdauer wird die Sperre automatisch aufgehoben. 🎜2.3 Legen Sie die Gültigkeitsdauer der Sperre fest
🎜Es ist sehr einfach, die Gültigkeitsdauer der Sperre festzulegen. Verwenden Sie einfach den Befehlexpire
Redis
, wie zum Beispiel: 🎜# lock为key,rand_id表示key里保存的值 redis-cli --eval unlock.lua lock , rand_id
expire
ausgeführt wird, dann expire< /code >Die Ausführung ist nicht erfolgreich und die Sperre wird nicht aufgehoben. Daher müssen wir sicherstellen, dass die beiden oben genannten Befehle zusammen ausgeführt werden. 🎜🎜Es gibt zwei Methoden: Eine besteht darin, ein in der Sprache <code>LUA
geschriebenes Skript zu verwenden, und die andere darin, den Befehl set
von Redis</code zu verwenden >, <code>set Nachdem dem Befehl
der Parameter nx
folgt, stimmt der Ausführungseffekt mit setnx
und dem set</ überein. Auf den Befehl code> kann der Parameter <code>ex
code> folgen, um die Ablaufzeit festzulegen. Daher können wir den Befehl set
verwenden, um setnx
und zusammenzuführen expire
zusammen, sodass die Ausführung garantiert atomar sein kann. 🎜rrreee🎜 hat das Problem effektiver Sperren gelöst, jetzt schauen wir uns ein anderes Problem an. 🎜🎜
A
, B
und C
alle Sperren erhalten, wenn sie einen bestimmten Vorgang ausführen, und die Sperren nach der Ausführung freigeben . 🎜现在的情况是进程A
执行第2步时卡顿了(上面绿色区域所示),且时间超出了锁有效期,所以进程A
设置的锁自动释放了,这时候进程B
获得了锁,并开始执行操作,但由于进程A
只是卡顿了而已,所以会继续执行的时候,在第3步的时候会手动释放锁,但是这个时候,锁由线程B
所拥有,也就是说进程A删除的不是自己的锁,而进程B的锁,这时候进程B
还没执行完,但锁被释放后,进程C
可以加锁,也就是说由于进程A卡顿释放错了锁,导致进程B和进程C可以同时获得锁。
怎么避免这种情况呢?如何区分其他进程的锁,避免删除其他进程的锁呢?答案就是每个进程在加锁的时候,给锁设置一个唯一值,并在释放锁的时候,判断是不是自己设置的锁。
2.4 给锁设置唯一值
给锁设置唯一值的时候,一样是使用set
命令,唯一的不同是将键值1改为一个随机生成的唯一值,比如uuid。
# rand_uid表示唯一id set lock rand_id nx ex 10
当锁里的值由进程设置后,释放锁的时候,就需要判断锁是不是自己的,步骤如下:
- 通过
Redis
的get
命令获得锁的值 - 根据获得的值,判断锁是不是自己设置的
- 如果是,通过
del
命令释放锁。
此时我们看到,释放锁需要执行三个操作,如果三个操作依次执行的话,是没有办法保证原子性的,比如进程A
在执行到第2步后,准备开始执行del
命令时,而锁由时有效期到了,被自动释放了,并被其他服务器上的进程B
获得锁,但这时候线程A
执行del
还是把线程B
的锁给删掉了。
解决这个问题的办法就是保证上述三个操作执行的原子性,即在执行释放锁的三个操作中,其他进程不可以获得锁,想要做到这一点,需要使用到LUA脚本。
2.5 通过LUA脚本实现释放锁的原子性
Redis
支持LUA
脚本,LUA
脚里的代码执行的时候,其他客户端的请求不会被执行,这样可以保证原子性操作,所以我们可以使用下面脚本进行锁的释放:
if redis.call("get",KEYS[1]) == ARGV[1] then return redis.call("del",KEYS[1]) else return 0 end
将上述脚本保存为脚本后,可以调用Redis
客户端命令redis-cli
来执行,如下:
# lock为key,rand_id表示key里保存的值 redis-cli --eval unlock.lua lock , rand_id
推荐学习:Redis视频教程
Das obige ist der detaillierte Inhalt vonEin Artikel, der ausführlich erklärt, wie man Redis zum Implementieren verteilter Sperren verwendet. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Heiße KI -Werkzeuge

Undresser.AI Undress
KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover
Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool
Ausziehbilder kostenlos

Clothoff.io
KI-Kleiderentferner

AI Hentai Generator
Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

Heiße Werkzeuge

Notepad++7.3.1
Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version
Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1
Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6
Visuelle Webentwicklungstools

SublimeText3 Mac-Version
Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Heiße Themen



Der Redis -Speicher steigt: zu großes Datenvolumen, unsachgemäße Datenstrukturauswahl, Konfigurationsprobleme (z. B. MaxMemory -Einstellungen zu klein) und Speicherlecks. Zu den Lösungen gehören: Löschen abgelaufener Daten, Verwendung der Komprimierungstechnologie, Auswahl geeigneter Strukturen, Anpassung der Konfigurationsparameter, der Überprüfung von Speicherlecks im Code und regelmäßiger Überwachung der Speicherverwendung.

Redis -Gedächtnisfragmentierung bezieht sich auf die Existenz kleiner freier Bereiche in dem zugewiesenen Gedächtnis, die nicht neu zugewiesen werden können. Zu den Bewältigungsstrategien gehören: Neustart von Redis: Der Gedächtnis vollständig löschen, aber den Service unterbrechen. Datenstrukturen optimieren: Verwenden Sie eine Struktur, die für Redis besser geeignet ist, um die Anzahl der Speicherzuweisungen und -freisetzungen zu verringern. Konfigurationsparameter anpassen: Verwenden Sie die Richtlinie, um die kürzlich verwendeten Schlüsselwertpaare zu beseitigen. Verwenden Sie den Persistenzmechanismus: Daten regelmäßig sichern und Redis neu starten, um Fragmente zu beseitigen. Überwachen Sie die Speicherverwendung: Entdecken Sie die Probleme rechtzeitig und ergreifen Sie Maßnahmen.

Starten Sie Redis neu, indem Sie die Konfigurationsdatei angeben: 1. Suchen Sie die Konfigurationsdatei (redis.conf normalerweise im Konfi -Unterverzeichnis); 2. Ändern Sie die erforderliche Konfiguration (z. B. das Ändern des Ports); 3.. Verwenden Sie den Befehl edis-server /path/to/redis.conf, um Redis über die Konfigurationsdatei neu zu starten (wobei /path/redis.conf der Pfad der geänderten Konfigurationsdatei ist); V.

Die Daten existieren immer noch nach dem Neustart von Redis. Redis speichert Daten im Speicher, und das Neustart löscht keine Speicherdaten. Redis liefert auch Persistenz, speichert Daten über RDB- oder AOF -Dateien auf der Festplatte auf der Festplatte und stellt sicher, dass Daten nach dem Neustart aus persistierenden Dateien wiederhergestellt werden können.

Die Wiederherstellung der Wiederherstellung nimmt zusätzlichen Speicher an, RDB erhöht vorübergehend die Speicherverwendung beim Generieren von Snapshots, und AOF nimmt beim Anhängen von Protokollen weiterhin Speicher auf. Einflussfaktoren umfassen Datenvolumen, Persistenzrichtlinien und Redis -Konfiguration. Um die Auswirkungen zu mildern, können Sie RDB -Snapshot -Richtlinien vernünftigerweise konfigurieren, die AOF -Konfiguration optimieren, die Hardware verbessern und Speicherverbrauch überwachen. Darüber hinaus ist es wichtig, ein Gleichgewicht zwischen Leistung und Datensicherheit zu finden.

Die Einstellung der Redis -Speichergröße muss die folgenden Faktoren berücksichtigen: Datenvolumen und Wachstumstrend: Schätzen Sie die Größe und Wachstumsrate gespeicherter Daten. Datentyp: Verschiedene Typen (z. B. Listen, Hashes) belegen einen anderen Speicher. Caching -Richtlinie: Vollständige Cache, teilweise Cache und Phasen -Richtlinien beeinflussen die Speicherverwendung. Business Peak: Verlassen Sie genug Speicher, um mit Verkehrsspitzen umzugehen.

So starten Sie den Redis-Dienst in verschiedenen Betriebssystemen neu: Linux/macOS: Verwenden Sie den SystemCTL-Befehl (SystemCTL Neustart REDIS-Server) oder den Servicebefehl (Service Redis-Server-Neustart). Windows: Verwenden Sie das Tool von Services.msc (Geben Sie "Services.msc" im Dialogfeld Ausführen und drücken Sie die Eingabetaste) und klicken Sie mit der rechten Maustaste auf den Dienst "Redis" und wählen Sie "Neustart".

Der Befehl rEDIS-Neustart ist Redis-Server. Dieser Befehl wird verwendet, um Konfigurationsdateien zu laden, Datenstrukturen zu erstellen, REDIS -Server zu starten und auf Client -Verbindungen zu hören. Benutzer können den Befehl "Redis-Server [Optionen]" im Terminal ausführen, um den Redis-Server neu zu starten. Zu den gemeinsamen Optionen gehören der Hintergrundvorgang, das Angeben von Konfigurationsdateipfad, das Angeben von Höranschluss und das Nachladen von Snapshots nur dann, wenn Daten verloren gehen. Beachten Sie, dass das Neustart des Servers alle Client -Verbindungen trennen und vor dem Neustart die erforderlichen Daten speichern.
