1、在 Swoole 中,这个 Redis 服务端是个什么东西呢?
Redis 服务端可不是我们说的去连接 Reids 服务器的那个东西,那个叫做 PHP 的 Reids 客户端。服务端的意思是一个可以提供服务的应用,redis-server 才是我们最熟悉的那个 Redis 的服务端。
那么在 Swoole 中,这个 Redis 服务端是个什么东西呢?其实它是一个基于 Redis 协议的服务器程序,可以让我们使用 Redis 的客户端来连接这个服务。光这么说估计大家还是一脸蒙圈,我们直接来看看效果再说。
use Swoole\Redis\Server;
define('DB_FILE', __DIR__ . '/db');
$server = new Server("0.0.0.0", 9501, SWOOLE_BASE);
if (is_file(DB_FILE)) {
$server->data = unserialize(file_get_contents(DB_FILE));
} else {
$server->data = array();
}
$server->setHandler('GET', function ($fd, $data) use ($server) {
if (count($data) == 0) {
return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'GET' command"));
}
$key = $data[0];
if (empty($server->data[$key])) {
return $server->send($fd, Server::format(Server::NIL));
} else {
return $server->send($fd, Server::format(Server::STRING, $server->data[$key]));
}
});
$server->setHandler('SET', function ($fd, $data) use ($server) {
if (count($data) < 2) {
return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'SET' command"));
}
$key = $data[0];
$server->data[$key] = $data[1];
return $server->send($fd, Server::format(Server::STATUS, "OK"));
});
$server->setHandler('sAdd', function ($fd, $data) use ($server) {
if (count($data) < 2) {
return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'sAdd' command"));
}
$key = $data[0];
if (!isset($server->data[$key])) {
$array[$key] = array();
}
$count = 0;
for ($i = 1; $i < count($data); $i++) {
$value = $data[$i];
if (!isset($server->data[$key][$value])) {
$server->data[$key][$value] = 1;
$count++;
}
}
return $server->send($fd, Server::format(Server::INT, $count));
});
$server->setHandler('sMembers', function ($fd, $data) use ($server) {
if (count($data) < 1) {
return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'sMembers' command"));
}
$key = $data[0];
if (!isset($server->data[$key])) {
return $server->send($fd, Server::format(Server::NIL));
}
return $server->send($fd, Server::format(Server::SET, array_keys($server->data[$key])));
});
$server->setHandler('hSet', function ($fd, $data) use ($server) {
if (count($data) < 3) {
return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'hSet' command"));
}
$key = $data[0];
if (!isset($server->data[$key])) {
$array[$key] = array();
}
$field = $data[1];
$value = $data[2];
$count = !isset($server->data[$key][$field]) ? 1 : 0;
$server->data[$key][$field] = $value;
return $server->send($fd, Server::format(Server::INT, $count));
});
$server->setHandler('hGetAll', function ($fd, $data) use ($server) {
if (count($data) < 1) {
return $server->send($fd, Server::format(Server::ERROR, "ERR wrong number of arguments for 'hGetAll' command"));
}
$key = $data[0];
if (!isset($server->data[$key])) {
return $server->send($fd, Server::format(Server::NIL));
}
return $server->send($fd, Server::format(Server::MAP, $server->data[$key]));
});
$server->on('WorkerStart', function ($server) {
$server->tick(10000, function () use ($server) {
file_put_contents(DB_FILE, serialize($server->data));
});
});
$server->start();
查看上面的代码,我们会发现这也是一个 Server 对象。然后主要是使用 setHandler() 方法来监听 Reids 命令,在这里我们看到了熟悉的 get、set 等命令的定义。然后我们指定了 $server->data ,可以将它看成是一个数据源,直接使用的就是一个文件,直接在当前测试环境目录下创建一个叫做 db 的空文件就可以了。在 setHandler() 方法中,我们使用 send() 方法来返回响应的命令信息,并通过 format() 方法格式化返回的响应数据。最后,通过一个监听一个 WorkerStart 事件,设置了一个每隔一秒将数据写入到 db 文件的操作。