0

0

mysql如何实现用户收藏列表_mysql关系设计实战

P粉602998670

P粉602998670

发布时间:2026-02-08 08:25:14

|

204人浏览过

|

来源于php中文网

原创

收藏功能必须用独立中间表user_favorites,含user_id、item_id、item_type(可选)字段,建联合唯一索引(user_id,item_id)及user_id单列索引,查收藏用JOIN而非子查询,取消收藏用DELETE而非软删除,高并发收藏用INSERT IGNORE防重复。

mysql如何实现用户收藏列表_mysql关系设计实战

收藏功能必须拆成独立表,不能加到用户或内容表里

直接在 users 表加个 favorite_ids 字段(比如存 CSV)或者在 articles 表加 favorited_by,短期看着省事,但很快会卡死:查某用户所有收藏、查某内容被多少人收藏、取消收藏时更新字符串都难做且慢。关系型数据库的强项是关联查询,不是字符串解析

正确做法是建一张中间表:user_favorites,至少包含三个字段:id(主键,非必需但建议)、user_id(外键指向 users.id)、item_id(指向被收藏的内容,比如 articles.idproducts.id),再加一个 item_type 字段支持多类型(可选,见下一点)。

  • 如果只收藏一种类型(比如全是文章),item_id 直接关联 articles.id,加联合唯一索引 (user_id, item_id) 防重复收藏
  • 如果要同时收藏文章、视频、商品,就用「泛型外键」:加 item_type(如 'article''video'),并确保应用层校验 item_id 确实存在对应记录——MySQL 本身无法对动态表名建外键
  • 别忘了给 user_id 单独建索引,否则按用户查收藏列表会全表扫描

查用户收藏列表时,JOIN 比子查询更稳

想查用户 ID=123 的所有收藏文章及标题,常见写法是:SELECT * FROM articles WHERE id IN (SELECT item_id FROM user_favorites WHERE user_id = 123)。这在数据量小的时候没问题,但一旦 user_favorites 有几十万行,MySQL 可能不走 user_id 索引,或生成临时表拖慢查询。

更可靠的是显式 JOIN

文赋Ai论文
文赋Ai论文

专业/高质量智能论文AI生成器-在线快速生成论文初稿

下载
SELECT a.* 
FROM user_favorites uf 
JOIN articles a ON uf.item_id = a.id 
WHERE uf.user_id = 123 
ORDER BY uf.id DESC 
LIMIT 20;
  • uf.id DESC 是按收藏时间倒序(假设用自增主键记录顺序),比用 created_at 更轻量;如果需要精确时间,就加 created_at 字段并索引它
  • 务必在 user_favorites(user_id, id) 上建联合索引,让排序和 LIMIT 能用上索引
  • 如果收藏量极大(比如用户收藏了 10 万条),考虑分页改用游标(cursor-based pagination),避免 LIMIT 100000, 20

取消收藏必须用 DELETE,不是 UPDATE

看到有人把 user_favorites 表设计成带 is_deleted 字段的软删除,这是错的。收藏本质是「存在性关系」,不是「状态记录」。软删除会让表越积越大,唯一约束失效(同一用户可重复插入已“删”的记录),查询逻辑变复杂。

  • 取消收藏就一条语句:DELETE FROM user_favorites WHERE user_id = 123 AND item_id = 456
  • 执行前不用先 SELECT 判断是否存在——直接 DELETE 返回影响行数,0 表示本来就没收藏,应用层据此返回提示即可
  • 确保该语句命中 (user_id, item_id) 联合索引,否则可能锁整张表(尤其在高并发取消场景)

注意事务边界和并发冲突

用户点「收藏」按钮时,典型流程是:先查是否已收藏 → 没有则插入。这个两步操作在并发下会出问题:两个请求几乎同时查,都得到「未收藏」,然后都插入,违反唯一约束。

  • 最简单解法:忽略唯一键冲突。用 INSERT IGNORE INTO user_favorites (user_id, item_id) VALUES (123, 456),冲突时静默失败,应用层按影响行数判断是否成功
  • 更严谨的做法是用 INSERT ... ON DUPLICATE KEY UPDATE,但这里不需要更新任何字段,所以 IGNORE 更轻量
  • 如果业务要求严格返回「本次是否新增」,那就得用事务 + SELECT ... FOR UPDATE 锁住用户行,但会降低并发能力,一般没必要

泛型外键(item_type)带来的校验责任完全落在应用层,数据库不兜底;而唯一索引和外键缺失的地方,就是线上出错的第一现场。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
mysql修改数据表名
mysql修改数据表名

MySQL修改数据表:1、首先查看数据库中所有的表,代码为:‘SHOW TABLES;’;2、修改表名,代码为:‘ALTER TABLE 旧表名 RENAME [TO] 新表名;’。php中文网还提供MySQL的相关下载、相关课程等内容,供大家免费下载使用。

672

2023.06.20

MySQL创建存储过程
MySQL创建存储过程

存储程序可以分为存储过程和函数,MySQL中创建存储过程和函数使用的语句分别为CREATE PROCEDURE和CREATE FUNCTION。使用CALL语句调用存储过程智能用输出变量返回值。函数可以从语句外调用(通过引用函数名),也能返回标量值。存储过程也可以调用其他存储过程。php中文网还提供MySQL创建存储过程的相关下载、相关课程等内容,供大家免费下载使用。

329

2023.06.21

mongodb和mysql的区别
mongodb和mysql的区别

mongodb和mysql的区别:1、数据模型;2、查询语言;3、扩展性和性能;4、可靠性。本专题为大家提供mongodb和mysql的区别的相关的文章、下载、课程内容,供大家免费下载体验。

283

2023.07.18

mysql密码忘了怎么查看
mysql密码忘了怎么查看

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql密码忘了怎么办呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

518

2023.07.19

mysql创建数据库
mysql创建数据库

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql怎么创建数据库呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

260

2023.07.25

mysql默认事务隔离级别
mysql默认事务隔离级别

MySQL是一种广泛使用的关系型数据库管理系统,它支持事务处理。事务是一组数据库操作,它们作为一个逻辑单元被一起执行。为了保证事务的一致性和隔离性,MySQL提供了不同的事务隔离级别。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

389

2023.08.08

sqlserver和mysql区别
sqlserver和mysql区别

SQL Server和MySQL是两种广泛使用的关系型数据库管理系统。它们具有相似的功能和用途,但在某些方面存在一些显著的区别。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

536

2023.08.11

mysql忘记密码
mysql忘记密码

MySQL是一种关系型数据库管理系统,关系数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。那么忘记mysql密码我们该怎么解决呢?php中文网给大家带来了相关的教程以及其他关于mysql的文章,欢迎大家前来学习阅读。

631

2023.08.14

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

39

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 2.2万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 824人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号