技术栈

主页 > 后端开发 >

Redis 通讯协议

redis命令发送格式:

*<参数数量> CRLF
$<参数 1 的字节数量> CRLF
<参数 1 的数据> CRLF
...
$<参数 N 的字节数量> CRLF
<参数 N 的数据> CRLF
其中CRLF表示 rn

举个例子:set name wuzhc
格式化输出:

*3
$3
set
$4
name
$5
wuzhc

说明:
  • *开头,表示有多少个参数,例如*3表示有3个参数(set, name, wuzhc)

  • $开头,表示参数的字节长度,例如$3表示set有3个字节,$4表示name有4个字节

  • 每行rn结尾

通信协议为:
*3
$3
set
$4
name
$5
wuzhc

Redis 回复

  • 状态回复(status reply)的第一个字节是 "+",例如+OK

  • 错误回复(error reply)的第一个字节是 "-",例如-No such key

  • 整数回复(integer reply)的第一个字节是 ":",例如:1

  • 批量回复(bulk reply)的第一个字节是 "$",例如 $5 wuzhc

  • 多条批量回复(multi bulk reply)的第一个字节是 "*",例如*2 $5 wuzhc $3r age

PHP 实现Redis客户端

<?php
/**
 * Created by PhpStorm.
 * User: [email protected]
 * Date: 2017年09月12日
 * Time: 9:08
 */

class Client
{
    private $_socket = null;

    public function __construct($ip, $port) 
    {
        $this->_socket = stream_socket_client(
            "tcp://{$ip}:{$port}",
            $errno,
            $errstr,
            1,
            STREAM_CLIENT_CONNECT
        );
        if (!$this->_socket) {
            exit($errstr);
        }
    }

    /**
     * 执行redis命令
     * @param $command
     * @return array|bool|string
     */
    public function exec($command)
    {      
        // 拼装发送命令格式
        $command = $this->_execCommand($command);

        // 发送命令到redis
        fwrite($this->_socket, $command);

        // 解析redis响应内容
        return $this->_parseResponse();
    }

    /**
     * 将字符改为redis通讯协议格式
     * 例如mget name age 格式化为 *3
$4
mget
$4
name
$3
age

     * @param $command
     * @return bool|string
     */
    private function _execCommand($command)
    {
        $line = '';
        $crlf = "
";
        $params = explode(' ', $command);
        if (empty($params)) {
            return $line;
        }

        // 参数个数
        $line .= '*' . count($params) . $crlf;

        // 各个参数拼装
        foreach ((array)$params as $param) {
            $line .= '$' . mb_strlen($param, '8bit') . $crlf;
            $line .= $param . $crlf;
        }

        return $line;
    }

    /**
     * 解析redis回复
     * @return array|bool|string
     */
    private function _parseResponse()
    {
        $line = fgets($this->_socket); 
        $type = $line[0]; 
        $msg = mb_substr($line, 1, -2, '8bit'); 

        switch ($type) {
            // 状态回复
            case '+':
                if ($msg == 'OK' || $msg == 'PONG') {
                    return true;
                } else {
                    return $msg;
                }
            // 错误回复
            case '-':
                exit($msg);
            // 整数回复
            case ':':
                return $msg;
            // 批量回复
            case '$': // $后面跟数据字节数(长度)
                $line = fread($this->_socket, (int)$msg + 2); // 数据字节数 + (
)两个字节
                return mb_substr($line, 0, -2, '8bit'); // 去除最后两个字节
            // 多条批量回复
            case '*': // *表示后面有多少个参数
                $data = [];
                for ($i = 0; $i < $msg; $i++) {
                    $data[] = $this->_parseResponse();
                }
                return $data;
        }
    }
}

// demo
$client = new Client('127.0.0.1', 6379);
$client->exec('set name wuzhc');
$res = $client->exec('get name');
var_dump($res);
责任编辑:admin  二维码分享:
本文标签: phpredis
点击我更换图片

评论列表