PHP Redis 处理WebSocket聊天记录的实例代码

具体代码如下所示:

<?php

ini_set('display_errors', 'on');

class chatClass
{
    private $redis;
    //这个变量模拟用户当前状态,是否登录,是否可查看
    public $checkUserReadable = false;

    //构造函数链接redis数据库
    function __construct()
    {
        $this->redis = new Redis();
        $this->redis->connect('127.0.0.1', '6379');
        $this->redis->auth('txlt');
    }

    /**
     * 发送消息时保存聊天记录
     * 这里用的redis存储是list数据类型
     * 两个人的聊天用一个list保存
     *
     * @from 消息发送者id
     * @to 消息接受者id
     * @meassage 消息内容
     *
     * 返回值,当前聊天的总聊天记录数
     */
    function setChatRecord($from, $to, $message)
    {
        $data = array('from' => $from, 'to' => $to, 'message' => $message, 'sent' => time()/*, 'recd' => 0*/);
        $value = json_encode($data);
        //生成json字符串
        $keyName = 'rec:' . $this->getRecKeyName($from, $to);
        //echo $keyName;
        $res = $this->redis->lPush($keyName, $value);
        if (!$this->checkUserReadable) { // 消息接受者无法立刻查看时,将消息设置为未读
            $this->cacheUnreadMsg($from, $to);
        }
        return $res;
    }

    /**
     * 获取聊天记录
     * @from 消息发送者id
     * @to 消息接受者id
     * @num 获取的数量
     *
     * 返回值,指定长度的包含聊天记录的数组
     */
    function getChatRecord($from, $to, $num)
    {
        $keyName = 'rec:' . $this->getRecKeyName($from, $to);
        //echo $keyName;
        $recList = $this->redis->lRange($keyName, 0, (int)($num));
        return $recList;
    }

    /**
     * 当用户上线时,或点开聊天框时,获取未读消息的数目
     * @user 用户id
     *
     * 返回值,一个所有当前用户未读的消息的发送者和数组
     * 数组格式为‘消息发送者id'=>‘未读消息数目'
     *
     */
    function getUnreadMsgCount($user)
    {
        return $this->redis->hGetAll('unread_' . $user);
    }

    /**
     * 获取未读消息的内容
     * 通过未读消息数目,在列表中取得最新的相应消息即为未读
     * @from 消息发送者id
     * @to 消息接受者id
     *
     * 返回值,包括所有未读消息内容的数组
     *
     */
    function getUnreadMsg($from, $to)
    {
        $countArr = $this->getUnreadMsgCount($to);
        $count = $countArr[$from];
        $keyName = 'rec:' . $this->getRecKeyName($from, $to);
        return $this->redis->lRange($keyName, 0, (int)($count));
    }

    /**
     * 将消息设为已读
     * 当一个用户打开另一个用户的聊天框时,将所有未读消息设为已读
     * 清楚未读消息中的缓存
     * @from 消息发送者id
     * @to 消息接受者id
     *
     * 返回值,成功将未读消息设为已读则返回true,没有未读消息则返回false
     */
    function setUnreadToRead($from, $to)
    {
        $res = $this->redis->hDel('unread_' . $to, $from);
        return (bool)$res;
    }

    /**
     * 当用户不在线时,或者当前没有立刻接收消息时,缓存未读消息,将未读消息的数目和发送者信息存到一个与接受者关联的hash数据中
     *
     * @from 发送消息的用户id
     * @to 接收消息的用户id
     *
     * 返回值,当前两个用户聊天中的未读消息
     *
     */
    private function cacheUnreadMsg($from, $to)
    {
        return $this->redis->hIncrBy('unread_' . $to, $from, 1);
    }


    /**
     * 生成聊天记录的键名,即按大小规则将两个数字排序
     * @from 消息发送者id
     * @to 消息接受者id
     *
     */
    private function getRecKeyName($from, $to)
    {
        return ($from > $to) ? $to . '_' . $from : $from . '_' . $to;
    }
}

/**
 *
 * 下面为测试用的代码 ,伪造数据模拟场景
 * 假定有两个用户id为2和3 ,2 向 3 发送消息
 *
 */
$chat = new chatClass();
$chat->checkUserReadable = true;
//$chat->checkUserReadable = get_cookie('username') ? true : false;

for ($i = 0; $i < 20; $i++) {
    # 发送消息时保存聊天记录
    $chat->setChatRecord('2', '3', 'message_' . $i);
}

echo '获得20条聊天记录: </br>';
# 获取聊天记录
$arr = $chat->getChatRecord('2', '3', 20);

for ($j = 0; $j < count($arr); $j++) {
    echo $arr[$j] . '</br>';
}

$chat->checkUserReadable = false;

for ($m = 0; $m < 5; $m++) {
    # 发送消息时保存聊天记录
    $chat->setChatRecord('2', '3', 'message_' . $m);
}

echo "</br>";

# 当用户上线时,或点开聊天框时,获取未读消息的数目
$umsg_1 = $chat->getUnreadMsgCount(3);

echo "未读消息计数: ";
//echo "</br>";
echo $umsg_1[2];
//print_r($umsg_1[2]);
echo "未读信息内容: </br> ";

# 获取未读消息的内容
$umsgContent = $chat->getUnreadMsg(2, 3);

for ($n = 0; $n < count($umsgContent); $n++) {
    echo $arr[$n] . '</br>';
}

echo "</br>";

# 将消息设为已读
$chat->setUnreadToRead(2, 3);

# 当用户上线时,或点开聊天框时,获取未读消息的数目
$umsg_2 = $chat->getUnreadMsgCount(3);

echo "未读消息计数: ";
//echo "</br>";
echo isset($umsg_2[2]) ? $umsg_2[2] : 0;
//print_r($umsg_2);

?>

测试代码打印结果如下所示:

获得20条聊天记录:
{"from":"2","to":"3","message":"message_19","sent":1631609088}
{"from":"2","to":"3","message":"message_18","sent":1631609088}
{"from":"2","to":"3","message":"message_17","sent":1631609088}
{"from":"2","to":"3","message":"message_16","sent":1631609088}
{"from":"2","to":"3","message":"message_15","sent":1631609088}
{"from":"2","to":"3","message":"message_14","sent":1631609088}
{"from":"2","to":"3","message":"message_13","sent":1631609088}
{"from":"2","to":"3","message":"message_12","sent":1631609088}
{"from":"2","to":"3","message":"message_11","sent":1631609088}
{"from":"2","to":"3","message":"message_10","sent":1631609088}
{"from":"2","to":"3","message":"message_9","sent":1631609088}
{"from":"2","to":"3","message":"message_8","sent":1631609088}
{"from":"2","to":"3","message":"message_7","sent":1631609088}
{"from":"2","to":"3","message":"message_6","sent":1631609088}
{"from":"2","to":"3","message":"message_5","sent":1631609088}
{"from":"2","to":"3","message":"message_4","sent":1631609088}
{"from":"2","to":"3","message":"message_3","sent":1631609088}
{"from":"2","to":"3","message":"message_2","sent":1631609088}
{"from":"2","to":"3","message":"message_1","sent":1631609088}
{"from":"2","to":"3","message":"message_0","sent":1631609088}
{"from":"2","to":"3","message":"message_4","sent":1631609065}

未读消息计数: 5未读信息内容:
{"from":"2","to":"3","message":"message_19","sent":1631609088}
{"from":"2","to":"3","message":"message_18","sent":1631609088}
{"from":"2","to":"3","message":"message_17","sent":1631609088}
{"from":"2","to":"3","message":"message_16","sent":1631609088}
{"from":"2","to":"3","message":"message_15","sent":1631609088}
{"from":"2","to":"3","message":"message_14","sent":1631609088}

未读消息计数: 0

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值