直接写入Mysql
直接写入Mysql是最简单的做法。
做两个表即可,
- post_like
记录文章被赞的次数,已有多少人赞过这种数据就可以直接从表中查到;
- user_like_post
记录用户赞过了哪些文章, 当打开文章列表时,显示的有没有赞过的数据就在这里面;
缺点
- 数据库读写压力大
热门文章会有很多用户点赞,甚至是短时间内被大量点赞, 直接操作数据库从长久来看不是很理想的做法。
redis直接存储
redis主要的特点就是快, 毕竟主要数据都在内存嘛;
优点
-
性能高
-
缓解数据库读写压力
其实我更多的在于缓解写压力, 真的读压力, 通过mysql主从甚至通过加入redis对热点数据做缓存都可以解决,写压力对于前面的方案确实是不大好使。
缺点
- 开发复杂
这个比直接写mysql的方案要复杂很多, 需要考虑的地方也很多;
- 不能保证数据安全性
不过对于我们点赞而已, 稍微丢失一点数据问题不大;
具体设计
redis设计部分:
post_set
在redis中弄一个set存放所有被点赞的文章
post_user_like_set_{$post_id}
对每个post以post_id作为key, 搞一个set存放所有对该post点赞的用户;
post_user_like_{$post_id}_{$user_id}
将每个用户对每个post的点赞情况放到一个hash里面去, hash的字段就
为啥用hash
只所以用hash是因为完全可以用hash来存储一个点赞的对象, 对应数据库的一行记录。
当然有同学会说用key, value也可以, 将所有的数据序列化(json_encode等)
后全部放到value里面去。 反复序列化也是一个很大的开销不是, hash可以很
方便的修改某个字段, 而序列化和反序列化的操作。
用户点赞/取消赞
获取user_id, post_id
, 查询该用户是否已经点过赞, 已点过就取消点赞,
将用户的点赞/取消赞的情况记录在redis中, 具体为:
- 写入
post_set
将post_id
写入post_set
- 写入
post_user_like_set_{$post_id}
将user_id
写入post_user_like_set_{$post_id}
- 写入
post_user_like_{$post_id}_{$user_id}
将用户点赞数据, 例如赞状态, post_id, user_id, ctime(操作时间), mtime(修改时间)写入post_user_like_{KaTeX parse error: Expected 'EOF', got '}' at position 8: post_id}̲_{user_id}中
- 更新
post_{$post_id}_counter
更新post_{$post_id}_counter,
这里的更新稍晚复杂一点, 需要和前面一样先获取当前用户是否对这个post点过赞
如果点过, 并且本次是取消赞, counter减一, 如果没点过, 本次是点赞, counter加一。
如果原来是取消赞的情况, 本次是点赞, counter加一。
使用Redis的有序集合可以做到根据关注的时间有序的取出列表,假设我的ID是me
,别人的ID是other
。
1、添加关注
- 将对方id添加到自己的关注列表中;
Redis::ZADD(“me:follow”, time(), other) - 将自己的id添加到对方的粉丝列表中:
Redis::ZADD(“other:fans”, time(), me)
2、取消关注
-
将对方id从自己的关注列表中移除;
Redis::ZREM(“me:follow”, other) -
将自己的id从对方的粉丝列表中移除:
Redis::ZREM(“other:fans”, me)
3、 关注列表
-
查看我的关注列表:
Redis::ZRANGE(“me:follow”, 0 , -1) -
查看别人的把id换掉就可以
Redis::ZRANGE(“other:follow”, 0 , -1)
4、 粉丝列表
-
查看我的粉丝列表:
Redis::ZRANGE(“me:fans”, 0 , -1) -
查看别人的把id换掉就可以
Redis::ZRANGE(“other:fans”, 0 , -1)
5、人物关系
- 我单向关注他
- 我的关注列表中有他(或他的粉丝列表中有我);
- 我的粉丝列表中没有他(或他的关注列表中没有我)。
Redis::ZSCORE(“me:fans”, other) #未返回分数
Redis::ZSCORE(“me:follow”, other) #返回分数
- 他单向关注我
- 我的关注列表中没有他(或他的粉丝列表中没有我);
- 我的粉丝列表中有他(或他的关注列表中有我)。
Redis::ZSCORE(“other:fans”, me) #未返回分数
Redis::ZSCORE(“other:follow”, me) #返回分数
- 是否互粉
- 我的关注列表中有他(或他的粉丝列表中有我);
- 我的粉丝列表中有他(或他的关注列表中有我)。同时成立才为互粉。
Redis::ZSCORE(“me:fans”, other) #返回分数
Redis::ZSCORE(“me:follow”, other) #返回分数
6、 数量相关
-
我的关注数
Redis::ZCARD(“me:follow”); #返回数量 -
我的粉丝数
Redis::ZCARD(“me:fans”); #返回数量
7、 排序取出所有的人
-
根据关注的时间倒叙取出用户的id
Redis::ZREVRANGE(“me:fans”, 0, -1, TRUE); #倒序取值 -
根据关注时间顺序取出用户的id
Redis::ZRANGE(“me:fans”, 0, -1, TRUE); #顺序取值
来源地址:https://blog.csdn.net/qq_45911678/article/details/122966183