Avec le développement d'Internet et de l'Internet mobile, la concurrence élevée et les systèmes distribués sont devenus des problèmes inévitables dans le développement quotidien. Dans ce cas, les verrous distribués deviennent un outil indispensable qui peut nous aider à éviter des problèmes tels que la concurrence entre les ressources et l’incohérence des données. Cet article explique comment implémenter des verrous distribués dans Swoole pour vous aider à mieux résoudre les problèmes de concurrence dans les systèmes distribués.
1. Qu'est-ce qu'un verrou distribué ?
Dans un système distribué, plusieurs processus accèdent aux ressources partagées en même temps. Afin de garantir que les données ne soient pas endommagées ou que des conflits de concurrence ne se produisent, ces ressources partagées doivent être verrouillées. Le verrou distribué est un mécanisme de verrouillage conçu pour permettre l'utilisation correcte des ressources partagées dans un système distribué.
La mise en œuvre des verrous distribués est relativement complexe. Généralement, les aspects suivants doivent être pris en compte :
2. Introduction à Swoole
Swoole est un moteur de communication réseau asynchrone et parallèle hautes performances pour le langage PHP. Côté serveur et côté client de divers protocoles tels que WebSocket. Les fonctionnalités de Swoole incluent :
3. Comment implémenter des verrous distribués dans Swoole ?
Ci-dessous, nous présenterons comment implémenter des verrous distribués dans Swoole.
Implémentation de verrous distribués basés sur RedisLe processus général d'utilisation de Redis pour implémenter des verrous distribués est le suivant :
(1) Obtenez un objet de connexion Redis via le pool de connexion Redis
(2) ) Utilisez la commande SETNX Pour réaliser l'exclusivité mutuelle du verrou, lorsque la valeur de retour est 1, cela signifie que l'occupation est réussie (3) Afin d'éviter un blocage, définissez le délai d'expiration du verrou # ; 🎜🎜# (4) Utilisez la commande DEL pour déverrouiller le verrou.
Ce qui suit est le code d'implémentation spécifique :
class RedisLock { private $redis; public function __construct($config) { $this->redis = new Redis(); $this->redis->connect($config['host'], $config['port'], $config['timeout']); if (!empty($config['auth'])) { $this->redis->auth($config['auth']); } } public function lock($key, $timeout = 10) { $startTime = time(); do { $result = $this->redis->setnx($key, time() + $timeout); if ($result) { return true; } $lockTime = $this->redis->get($key); if ($lockTime && $lockTime < time()) { $oldTime = $this->redis->getset($key, time() + $timeout); if ($oldTime == $lockTime) { return true; } } usleep(100); // 100毫秒等待 } while (time() - $startTime < $timeout); return false; } public function unlock($key) { $this->redis->del($key); } }Dans le code ci-dessus, la fonction de verrouillage utilise une boucle do-while pour attendre que le verrou soit libéré lorsque le. le temps d'attente dépasse le délai indiqué. Lorsque le délai d'attente se produit, false est renvoyé ; la commande DEL est utilisée dans la fonction de déverrouillage pour libérer le verrou. Bien que cette méthode soit simple à mettre en œuvre et entraîne une faible surcharge, elle présente également une certaine probabilité de blocage.
Implémentation de verrous distribués basés sur Zookeeper
(3) Récupère tous les nœuds dans Zookeeper et les trie par numéro de série de nœud
(4) Compare son propre numéro de série de nœud avec le numéro de série de ; le plus petit nœud actuel, s'ils sont égaux, alors indique que le verrou a été obtenu, sinon il écoutera le dernier nœud avec un numéro de série plus petit que le sien (5) Lorsqu'un nœud avec un numéro de série plus petit ; Un numéro supérieur au sien est supprimé, le nœud actuel reçoit une notification d'événement, puis répète la quatrième étape.
Ce qui suit est le code d'implémentation spécifique :
class ZookeeperLock { private $zk; private $basePath = '/lock'; private $myNode; public function __construct($config) { $this->zk = new Zookeeper(); $this->zk->connect($config['host'] . ':' . $config['port']); if (isset($config['auth'])) { $this->zk->addAuth('digest', $config['auth']); } if (!$this->zk->exists($this->basePath)) { $this->zk->create($this->basePath, null, array(array('perms' => Zookeeper::PERM_ALL, 'scheme' => 'world', 'id' => 'anyone')), null); } } public function lock() { $this->myNode = $this->zk->create($this->basePath . '/node_', null, array(array('perms' => Zookeeper::PERM_ALL, 'scheme' => 'world', 'id' => 'anyone')), Zookeeper::EPHEMERAL | Zookeeper::SEQUENCE); while (true) { $children = $this->zk->getChildren($this->basePath); sort($children); $pos = array_search(basename($this->myNode), $children); if ($pos === 0) { return true; } else { $this->zk->exists($this->basePath . '/' . $children[$pos - 1], function ($event_type, $s, $event_data) { $this->unlock(); }); usleep(100); // 100毫秒等待 } } } public function unlock() { if ($this->myNode) { $this->zk->delete($this->myNode); $this->myNode = null; } } }
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!