点赞是个很常见的功能,以前我在eblog中使用的点赞是先在redis中增长,然后定时同步数据库。那么除了这个还有其他方案吗?那你知道如何点赞去重吗?看看这篇文章。

点赞功能是目前 app 开发基本的功能

今天我们就来聊聊 点赞、评论、收藏等这些场景的 db 数据库设计问题,

1. 我们先来看看场景的需求:

a. 显示点赞数量

b. 判断用户是否点过赞,用于去重,必须的判断

c. 显示个人点赞列表,一般在用户中心

d. 显示文章点赞列表

我们先看一下头条和微博的例子

头条的点赞

微博的点赞

这两个都是具有顶级流量的,后端肯定有复杂的架构,我们今天只谈大众化的方案。

2.1 mysql 方案

mysql 方案, 随着 nosql 的流行,大数据的持续热点,但是 mysql 仍然不可替代,对于大多数的中小项目,低于千万级的数据量,采用 mysql 分表 + cache,是完全可以胜任的,而且稳定性是其他方案无可比拟的:

文章表create table post {post_id int(11) NOT NULL AUTO_INCREMENT,......star_num int(11) COMMENT '点赞数量'
}用户表create table user {user_id int(11) NOT NULL AUTO_INCREMENT,......star_num int(11) COMMENT '点赞数量'
}点赞表create table star {id int(11) NOT NULL AUTO_INCREMENT,post_id,user_id,......
}

常用的查询:

查询用户点赞过的文章 select post_id from star where user_id=?查询文章的点赞用户 select user_id from star where post_id=?

点赞数量可以通过定时异步统计更新到 post 和 user 表中。

数据量不大的时候,这种设计基本可以满足需求了,

缺点:

数据量大时,一张表在查询时压力巨大,需要分表,而不论用 postid 还是 userid 来 hash 分表都与我们的需求有冲突,唯一的办法就是做两个表冗余。这增加了存储空间和维护工作量,还可能有一致性问题。

2.2 redis 方案

当数据量达到上亿的量,上 cache 是必经的阶段,由于点赞这种动作很随意,很多人看到大拇指就想点,所以数据量增长很快,数据规模上来后,对 mysql 读写都有很大的压力,这时就要考虑 memcache、redis 进行存储或 cache。

为什么一般都选择 redis, redis 作为流行的 nosql,有着丰富的数据类型,可以适应多个场景的需求。

采用 redis 有两种用途,一种是 storage,一种是纯 cache,需要 + mysql 一起。纯 cache 就是把数据从 mysql 先写入 redis,用户先读 cache,miss 后再拉取 MySQL,同时 cache 做同步。

cache

多数场景二者是同时使用的,并不冲突。

下面说下 redis 作为 storage 的方案:

场景 a :显示点赞数量

在点赞的地方,只是显示一个点赞数量,能区分用户是否点赞过,一般用户不关心这个列表,这个场景只要一个数字就可以了,当数量比较大时,一般显示为 "7k" ,"10W" 这样。

以文章 id 为 key

//以文章id=888为例
127.0.0.1:6379[2]> set star:tid:888 898 //设置点赞数量
OK
127.0.0.1:6379[2]> incr star:tid:888 //实现数量自增 (integer)
899

场景 b:点赞去重,避免重复点赞

要实现这个需求,必须有文章点赞的 uid 列表,以 uid 为 key 场景 c:一般在用户中心,可以看到用户自己的点赞列表

阅读:Github上最值得学习的100个Java开源项目,涵盖各种技术栈!

这个需求可以使用场景 b 的数据来实现。

用户中心点赞列表

场景 d:文章的点赞列表,类似场景 b,以文章 id 为 key

//以文章id=888为例
127.0.0.1:6379[2]> sadd star:list:tid:888 123 456 789 //点赞uid列表 (integer)
3
127.0.0.1:6379[2]> sismember star:list:tid:888 456 //判断是否点赞 (integer)
1

点赞的地方,如果点赞过显示红色,没有则显示黑白色,

今日头条是没有地方可以看到点赞列表的,而微博点进去,详情页可以看到点赞列表,但是只会显示最近的几十条,没有分页显示。

如下图,我选了一条热点,拥有众多粉丝的 “猪猪”

帖子点赞列表

可能有人觉得,点赞列表没人关心,存储又会浪费大量资源,不如不存!但是,这个数据是必须要有的。两点:

a. 去重。点赞数可以不精确,但去重必须是精确的,

b. 另外一个社交产品,用户行为的一点一滴都需要记录,对于后续的用户行为分析和数据挖掘都是有意义的。

上面使用 string 存储的用户点赞数量,除了 string,还可以用 hash 来存储,对文章 id 分块,每 100 个存到一个 hash,分别存入 hash table,每个文章 id 为 hash 的一个 key,value 存储点赞的用户 id,如果点赞用户很多,避免 id 过多产生性能问题,可以单列出来,用 sorted set 结构保存,热点的毕竟是少数。

hash

方案优缺点比对

hash:使用了更少的全局 key ,节省了内存空间;但是也带来了问题

如何根据文章 id 路由到对应的 hash?

查找一个用户 id 是在 hash 还是 set?存在不确定性

使用 hash 虽然节省了空间,但增加了复杂度,如何选择就看个人需求了。

除此之外,你还有其他的方法吗?

3. 数据一致性

redis 作为 storage 使用时,一定要做好数据的持久化,必须开启 rdb 和 aof,这会导致业务只能使用一半的机器内存,所以要做好容量的监控,及时扩容。

另外只要有数据 copy,就会有一致性问题,这就是另外一个很重要的话题了。

mysql 查询列表是否关注_点赞功能,用mysql还是redis?相关推荐

  1. mysql查询第八页_第八节:MySQL之Select指令详解和相关练习

    一. 前言 该篇文章基于之前   https://www.cnblogs.com/yaopengfei/p/7182230.html  的基础上进行补充修改. 1. 简介 就查询而言,可以简单的分为: ...

  2. MySQL查询不支持中文_为什么我的 Mysql 不支持中文查询?

    Q: 我在写一个查询条件时的问题如下: 如我想写一个字段中包含"李"字的所有记录 $str="李"; select * from table where fie ...

  3. mysql查询结果输出文件_如何将MySQL查询输出保存到文件?

    mysql查询结果输出文件 We can use the MySQL outfile statement to save the query output into a file. This is v ...

  4. mysql 去重_点赞功能,用 MySQL 还是 Redis ?

    点赞功能是目前app开发基本的功能 今天我们就来聊聊点赞.评论.收藏等这些场景的db数据库设计问题~ 1. 我们先来看看场景的需求: 显示点赞数量 判断用户是否点过赞,用于去重,必须的判断 显示个人点 ...

  5. mysql 查询和修改组合_## 超详细MySQL常用语句,增删查改

    ## MySQL常用语句,增删查改,安装配置mysql服务 ***#新手博客,应届生,谢谢支持哟 记得点赞关注哟*** ***-----sql常见命令:--------*** 安装服务:mysqld ...

  6. php中mysql查询条件为数组_请教php中数组做为mysql查询条件wherexxinarray,怎么处理...

    原先想着把数组 $customerid 拆分出来,判断长度给不同的变量,变量多少还要再判断,然后写 下边的mysql查询,有些麻烦,请教有没有好的方法,谢了! -------------------- ...

  7. mysql查询时调用函数_【性能测试】性能测试时对Mysql数据库的慢查询监控

    在做Web项目性能测试过程中,肯定要对数据库SQL语句执行情况实施监控,以便给开发提供准确的性能优化意见.目前Mysql数据库可以说是使用最广泛的数据库了,接下来咱们谈一下怎么使用Mysql数据库提供 ...

  8. mysql查询并设置高亮_慢查询分析调优工具~mysqldumpslow

    在日常的业务开发中,MySQL出现慢查询是很常见的,要么说明你家产品的增长性很好,要么就是你的SQL写的太烂了.所以对慢查询SQL进行分析和优化很重要,其中mysqldumpslow是MySQL服务自 ...

  9. mysql查询语句详解_基于mysql查询语句的使用详解

    1> 查询数据表除了前三条以外的数据. 起初我想到的是这条语句 SELECT * FROM admin WHERE userid NOT IN (SELECT userid FROM admin ...

最新文章

  1. 6大准则+10道习题,终于有人把怎样选择图表讲明白了
  2. Redis Labs 再次更改开源许可证,但 Redis 本身不受影响
  3. elasticsearch 写入数据并查询
  4. 使SSH不用输入密码
  5. P4819-[中山市选]杀人游戏【tarjan】
  6. linux操作系统下部署Javaweb项目教程
  7. JavaScript数据结构——树(Tree)
  8. 实训|第十一天学习一下linux中的进程,文件查找,文件压缩与IO重定向
  9. BZOJ 2724: [Violet 6]蒲公英
  10. 3. SQL 语句本身的优化(慢查询)
  11. OpenCV(iOS)平滑处理(模糊,毛玻璃)(10)
  12. 企业选择WMS仓库管理系统免费版是否更好
  13. 【STM32H7的DSP教程】第7章 ARM DSP源码和库移植方法(IAR8)
  14. APP自动化测试-3. Appium元素定位与等待
  15. 远程手机控制开关应用
  16. Python3.6爬取前程无忧
  17. 电机速度曲线规划_S曲线
  18. c4d安装完 只有语言英文的,为何我将显示语言设置成英文后开始菜单分类还依然显示为中文?...
  19. 大神教你如何优化变压器匝间电容?
  20. ubuntu 生成桌面快捷方式(有图)

热门文章

  1. 二叉树——美国血统(洛谷 P1827)
  2. 实用教程丨官方下载Oracle各版本安装软件及补丁包
  3. openGauss 上海 Meetup:把企业级数据库能力带给用户
  4. 倒计时7天:招行、平安齐聚DTC 2019之金融峰会,讲师议题抢先看
  5. 一篇文章带你读懂 MySQL 和 InnoDB
  6. “七大属性加持,三个全新升级组件”这个高性能利器有点厉害
  7. 从云数据迁移服务看MySQL大表抽取模式
  8. 软件工程的迷途和沉思
  9. 【华为云技术分享】“技术-经济范式”视角下的开源软件演进剖析-part 1
  10. 【华为云技术分享】#探索鲲鹏#之“在鲲鹏上使用编程语言——C语言