Swoole系列2.6Redis 服务器
Posted 码农老张Zy
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Swoole系列2.6Redis 服务器相关的知识,希望对你有一定的参考价值。
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 文件的操作。
客户端测试
现在,运行起来这个 Swoole 写的 Redis 服务端文件吧,然后使用你的 redis-cli 来连接并测试它。
神奇吗?惊喜吗?我们竟然实现了一个 Redis 服务器,再看看 db 文件中是什么内容。
不出所料,db 中保存的是我们序列化之后的内容。这个东西是不是很有意思,完全可以做一个我们自己的小 Redis 来用。对于一些小网站,小应用来说,你不需要再去安装一个庞大的 Redis 服务了,直接使用 Swoole 就可以实现一个遵循 Redis 协议规范的小型缓存服务器了。
添加一个命令
话说回来,注意到上面测试的截图中我们使用了 keys 命令却返回了一个 unknow command 吗?因为在这里,所有的命令我们都要在 setHandler() 中一个一个的定义。相对来说,这东西写起来的还是比较麻烦的。最后,我们再来简单地实现一个 keys 命令好了。
$server->setHandler('KEYS', 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"));
if(!$server->data)
return $server->send($fd, Server::format(Server::NIL));
$key = $data[0];
if($key == "*")
return $server->send($fd, Server::format(Server::SET, array_keys($server->data)));
else
$dataKeys = array_keys($server->data);
$key = str_replace(["*", "?"], [".*", ".?"], $key);
$values = array_filter($dataKeys, function($value) use ($key)
return preg_match('/'.$key.'/i', $value, $mchs);
);
if($values)
return $server->send($fd, Server::format(Server::SET, $values));
return $server->send($fd, Server::format(Server::NIL));
);
并不是很规范,但功能是没问题的,大家可以测试一下效果。
总结
关于 Redis 服务器这一块的内容,我们了解一下就好了,实际的使用中自己去写这些东西还是挺费劲的。到这里为止,我们的入门相关课程就学习完了。
大家还记得讲了什么吗?就是简单地搭起了 Http/TCP/UDP/WebSocket/Redis 这几个服务。通过这几个服务,我们日常的 Web 应用其实就已经全部都可以覆盖了。只是说,我们都是使用的原生的 Swoole 对象及函数来实现的非常简单的测试代码。真正在实际的项目中使用的话,一般还是会借助框架来实现。
那么我们就继续进阶方面的内容,到最后才会简单地学习了解一个 Swoole 框架。别急,基础才是王道嘛!
测试代码:
https://github.com/zhangyue0503/swoole/blob/main/2.%E5%9F%BA%E7%A1%80/source/2.6Redis%E6%9C%8D%E5%8A%A1%E5%99%A8.php
参考文档:
https://wiki.swoole.com/#/redis_server
以上是关于Swoole系列2.6Redis 服务器的主要内容,如果未能解决你的问题,请参考以下文章
Swoole+Redis+webSocket实现点对点即时聊天