PHP+redis实现微博的拉模型案例详解


Posted in PHP onJuly 10, 2019

本文实例讲述了PHP+redis实现微博的拉模型。分享给大家供大家参考,具体如下:

上回写了一篇推模型的内容,这回分享一篇拉模型的内容。

拉模型

拉模型就是展示微博的时候,获取自己的所有关注的人,然后从关注的人中拉取最新微博。

微博项目数据结构设计

user表设计

注册的时候将user数据写入redis中,key如下:

user数据的key
用户名=user:uesrid:$uesrid:username
密码=user:userid:$userid:password

还需要这样写一份,因为需要靠用户名来登录,这样就可以根据用户名来查询用户id。

user:username:userid:$userid

关注的人和粉丝设计

每个用户在产生关注的动作后,在redis中维护两个无序集合set,一个是following,一个是follower,following集合保存的是我关注的人,follower集合保存的是我的粉丝。注意是每个用户都要维护这样的两个集合,用userid来区别。

单条微博表设计

每条微博的信息用hash结构来存储,根据不同的微博id来区分,每条微博有如下信息:发布人id,发布人昵称,发布时间,微博内容。

拉取关注者微博表 设计

每个用户发布微博后,维护20条最新微博,并保存到有序集合sorted set中,用不同的userid来区分。

注意:有序集合的score用微博id,集合保存的也是微博id。

个人微博表

每个用户维护自己的微博,保存到链表中,只保存1000条,redis中只保存1000条微博数据,如果想查询更多,去数据库中查询。

个人已拉取表设计

每个用户在拉取微博后,将微博保存到已经拉取的表中,这个表是一个链表结构,最多保存1000条微博。

发布微博

首先将微博保存成hash结构,然后将微博保存到拉取表,还保存到个人微博表。

//1、保存微博
$conn = connredis();
$postid = $conn->incr('global:postid');
$conn->hmset('post:postid:'.$postid,['userid'=>$user['userid'],'username'=>$user['username'],'time'=>time(),'content'=>$content]);
//2、每个人维护20条最新微博,保存到有序集合中
$conn->zadd('starpost:userid:'.$user['userid'],$postid,$postid);
if($conn->zcard('starpost:userid:'.$user['userid']) > 20){
  $conn->zremrangebyrank('starpost:userid:'.$user['userid'],0,0);
}
//3、维护个人的1000条微博,保存到链表中
$conn->lpush('mypost:userid:'.$user['userid'],$postid);
if($conn->llen('mypost:userid:'.$user['userid']) > 1000){
  $conn->rpoplpush('mypost:userid:'.$user['userid'],'global:post');
}

展示微博

首先获取所有关注的人,获取上次拉取微博的位置,根据上次拉取的微博位置来拉取数据。然后给微博排序,设置新的拉取的位置,写入到已拉取表中,获取微博的详细内容,最后获取粉丝和关注数。进行展示即可。

//1、获取拉取对象
$stars = $conn->smembers('following:'.$user['userid']);//获取所有关注的人
$stars[] = $user['userid'];//需要拉取自己的微博
//2、获取上次拉取的位置
$lastpull = $conn->get('lastpull:userid:'.$user['userid']);
if(!$lastpull){
$lastpull = 0;
}
//3、拉取微博 
$latest = [];
foreach($stars as $star){
$latest = array_merge($latest,$conn->zrangebyscore('starpost:userid:'.$star,$lastpull+1,1<<32-1));
}
//4、给微博排序
sort($latest,SORT_NUMERIC);
//5、设置拉取的位置
if(!empty($latest)){
  $conn->set('lastpull:userid:'.$user['userid'],end($latest));
}
//6、写入到已拉取表中
foreach($latest as $l){
  $conn->lpush('receivepost:'.$user['userid'],$l);
}
$conn->ltrim('receivepost:'.$user['userid'],0,999);//至多显示1000条微博
//7、获取微博的详细内容
$postids = $conn->sort('receivepost:'.$user['userid'],['sort'=>'desc']);
$posts = [];
foreach($postids as $postid){
  $posts[] = $conn->hmget('post:postid:'.$postid,['userid','username','time','content']);
}
//8、获取粉丝和关注数
$fansnum = $conn->scard('follower:'.$user['userid']);
$follownum = $conn->scard('following:'.$user['userid']);

Q&A

如何保证拉取的数据时最新的?

在拉取的时候,将最近拉取的微博id保存到redis中,然后下次我们只需要去拉取比这次保存的微博id大的微博,就可以保证拉取的数据是之前没有拉取的。

如何拉取所有关注者的数据?

遍历关注者,然后拉取数据

假设拉取A关注者的微博1,4,5 B关注者2,3,但是他们的发布时间交错,怎么展示数据?

将所有关注者的最新微博都取出来,然后根据微博id进行排序。

希望本文所述对大家PHP程序设计有所帮助。

PHP 相关文章推荐
初探PHP5
Oct 09 PHP
在PHP中读取和写入WORD文档的代码
Apr 09 PHP
php cookies中删除的一般赋值方法
May 07 PHP
PHP中使用strpos函数实现屏蔽敏感关键字功能
Aug 21 PHP
php ImageMagick windows下安装教程
Jan 26 PHP
php获取系统变量方法小结
May 29 PHP
7个鲜为人知却非常实用的PHP函数
Jul 01 PHP
PHP处理会话函数大总结
Aug 05 PHP
使用PHP如何实现高效安全的ftp服务器(二)
Dec 30 PHP
mysqli扩展无法在PHP7下升级问题的解决
Sep 10 PHP
使用PHP开发留言板功能
Nov 19 PHP
PHP文件打开关闭及读写操作示例解析
Aug 06 PHP
php解压缩zip和rar压缩包文件的方法
Jul 10 #PHP
PHP+redis实现微博的推模型案例分析
Jul 10 #PHP
实例分析10个PHP常见安全问题
Jul 09 #PHP
PHP批斗大会之缺失的异常详解
Jul 09 #PHP
PHP结合Redis+MySQL实现冷热数据交换应用案例详解
Jul 09 #PHP
PHP+Redis开发的书签案例实战详解
Jul 09 #PHP
使用composer命令加载vendor中的第三方类库 的方法
Jul 09 #PHP
You might like
编写PHP脚本来实现WordPress中评论分页的功能
2015/12/10 PHP
PHP Post获取不到非表单数据的问题解决办法
2018/02/27 PHP
jquery 入门教程 [翻译] 推荐
2009/08/17 Javascript
在Ajax中使用Flash实现跨域数据读取的实现方法
2010/12/02 Javascript
jquery 单引号和双引号的区别及使用注意
2013/07/31 Javascript
js中的onchange和onpropertychange (onchange无效的解决方法)
2014/03/08 Javascript
jquery css 设置table的奇偶行背景色示例
2014/06/03 Javascript
jQuery获取节点和子节点文本的方法
2014/07/22 Javascript
jQuery使用之标记元素属性用法实例
2015/01/19 Javascript
js获取图片宽高的方法
2015/11/25 Javascript
javascript html实现网页版日历代码
2016/03/08 Javascript
jquery实现焦点轮播效果
2017/02/23 Javascript
react native 文字轮播的实现示例
2018/07/27 Javascript
node中的session的具体使用
2018/09/14 Javascript
Python中实现两个字典(dict)合并的方法
2014/09/23 Python
python删除列表中重复记录的方法
2015/04/28 Python
详细解读Python中解析XML数据的方法
2015/10/15 Python
selenium+python自动化测试之多窗口切换
2019/01/23 Python
Python实现微信消息防撤回功能的实例代码
2019/04/29 Python
一篇文章弄懂Python中的可迭代对象、迭代器和生成器
2019/08/12 Python
python2爬取百度贴吧指定关键字和图片代码实例
2019/08/14 Python
PyCharm搭建Spark开发环境的实现步骤
2019/09/05 Python
Python如何实现后端自定义认证并实现多条件登陆
2020/06/22 Python
用python写爬虫简单吗
2020/07/28 Python
HTML5 canvas基本绘图之绘制五角星
2016/06/27 HTML / CSS
ECHT官方网站:男女健身服
2020/02/14 全球购物
DOUGLAS波兰:在线销售香水和化妆品
2020/07/05 全球购物
Overload和Override的区别。Overloaded的方法是否可以改变返回值的类型
2013/10/30 面试题
信息工程学院毕业生推荐信
2013/11/05 职场文书
趣味运动会策划方案
2014/06/02 职场文书
励志演讲稿800字
2014/08/21 职场文书
开展党的群众路线教育实践活动个人对照检查材料
2014/11/05 职场文书
2015国庆节66周年演讲稿
2015/03/20 职场文书
2016中秋节问候语
2015/11/11 职场文书
各类场合主持词开场白范文集锦
2019/08/16 职场文书
nginx请求限制配置方法
2021/07/09 Servers