zoukankan      html  css  js  c++  java
  • php redis 处理websocket聊天记录

    <?php
    ini_set('display_errors', 'on');
    
    class chatClass {
        private $redis;
    
        //这个变量模拟用户当前状态,是否登录,是否可查看
        public $checkUserReadable = false;
    
        //构造函数链接redis数据库
        public function __construct() {
            $this -> redis = new Redis();
            $this -> redis -> connect('127.0.0.1', '6379');
            $this -> redis -> auth('***cnblogs.com/handle');
        }
    
        /*
        发送消息时保存聊天记录
        * 这里用的redis存储是list数据类型
        * 两个人的聊天用一个list保存
        *
        * @from 消息发送者id
        * @to 消息接受者id
        * @meassage 消息内容
        *
        * 返回值,当前聊天的总聊天记录数
        */
        public 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 获取的数量
        *
        * 返回值,指定长度的包含聊天记录的数组
        */
        public 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’=>‘未读消息数目’
        *
        */
        public function getUnreadMsgCount($user) {
            return $this -> redis -> hGetAll('unread_' . $user);
        }
    
        /*
        * 获取未读消息的内容
        * 通过未读消息数目,在列表中取得最新的相应消息即为未读
        * @from 消息发送者id
        * @to 消息接受者id
        *
        * 返回值,包括所有未读消息内容的数组
        *
        *
        */
        public 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
        */
    
        public 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;
    for ($i = 0; $i < 20; $i++) {
    $chat -> setChatRecord('2', '3', 'message_' . $i);
    }
    
    echo 'get 20 chat records</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 "Unread message counts ";
    echo "</br>";
    print_r($umsg_1);
    echo "Unread message content </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 "</br>";
    echo "Unread message counts ";
    echo "</br>";
    print_r($umsg_2);
    *
    */
    ?>
  • 相关阅读:
    SRM 551 div2
    HDU_4390 Number Sequence (容斥原理)
    HDU 多校联合第五场
    HDU 多校联合第六场
    POJ 2057 The Lost House (经典树形dp)
    Lucas定理
    HDU 4385 Moving Bricks (状态dp+贪心)
    HDU 多校联合第三场
    当最短路变成二维 _!
    POJ 1848 (一道不错的树形dp)
  • 原文地址:https://www.cnblogs.com/yszr/p/10790612.html
Copyright © 2011-2022 走看看