作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO
联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬
上一篇提到树形结构是非常经典的一种表设计模式,看似平平无奇,实则包罗万象。今天,我们借助“用户评论”的需求,再来领略一把树形结构的魅力。
二级评论与盖楼
下面是两张评论相关的截图,请大家观察一下结构上有什么不同:
第一张图的评论形式俗称“二级评论”,第二张图俗称“盖楼”。
“二级评论”和“盖楼”最大的不同是 :
二级评论只需要关注当前评论的上一级,而“盖楼”则需要把当前评论之前的所有评论按顺序展示出来。
没有太多表设计经验的同学可能已经晕了:我上面两张图啥区别都还没整明白呢,被你这么一说,更晕了。
我们逐个分析。
先看“盖楼”。我们把“我是煎饼侠”的上一条评论也显示出来:
“我是煎饼侠”的评论其实是对“bravo1988”评论的评论,但它并没有直接回复“bravo1988”,而是另起一层并把前面的评论引用过来,然后在最下面显示自己的评论内容。
再看“二级评论”:
针对“谢函”的“想咨询一个...”的评论,“程大治”是直接在该评论下显示自己的评论内容,并不会另起一层(用户Shayne_xxy那种才叫另起一层)。整个评论区的所有评论其实就两大类:一级评论、二级评论。
关于“一级评论”、“二级评论”的定义:
- 一级评论:针对内容(文章、图片、视频)本身的评论
- 二级评论:针对一级评论的回复,也就是“对评论的评论”
上图中,“想咨询一个...”和“这个好”属于一级评论,“程大治”的两条评论属于二级评论(大家移上去再看一遍)。一级评论下无论再怎么复杂(A评论B,B评论A,C评论B...),都是二级评论,而不是三级评论、四级评论。
如果把上图改为“盖楼”,就是这样:
“盖楼”和“二级评论”两种评论形式看起来好像大相径庭,其实数据库表设计是差不多的,区别在SQL查询以及前端展示,其中“盖楼”的难度要大一些。
大家也看到了,“盖楼”这种形式的评论不如“二级评论”来得直观,且实现较为复杂(每一条评论都要找到在它前面的所有评论,d评论找c评论,c评论找b评论,最终找到a评论),所以现在已经很少采用 “盖楼”的评论形式了。
本文主要讨论如何设计“二级评论表”。
分析需求,确定表字段
请大家停下来重新观察并思考:如果让你来做这个需求,你会如何设计表结构,后端大概需要返回哪些字段呢?
需要几张表?
我们最直观的感受是:既然评论总共有2级,那么我们设计两张表吧,用逻辑外键关联一级评论表和二级评论表。
但实际上,这个问题可以用一张表“自关联”解决,只需要在表中设计一个pid,让secondLevel.pid = firstLevel.id即可。
所以,结论是二级评论可以用一张表解决。
一级评论和二级评论怎么摆放?
这个问题,其实是从前端展示的角度提出来的。我们知道,数据库存放的评论都是一条条独立的:
怎么最终在前端展示成这样呢:
还是树形结构,“二级评论”的树只有两级:
归根到底,用户评论这个需求还是对树形结构的实际应用。二级评论可以通过pid找到自己所属的一级评论,页面展示时,先遍历一级评论,直接展示在文章下方,再遍历一级评论的子评论(replies),把二级评论展示在一级评论下方即可。至于某个一级评论下的排序,可以默认id排序(一般等于时间排序)。如果有其他需要,可自定排序规则。
张三@李四
现在只剩最后一个问题了,怎么处理张三@李四这种展示效果?比如下图:
无论一级评论还是二级评论,除了展示评论本身,还要展示评论相关的用户信息、评论时间等。
而二级评论还多了一个属性:这条评论是谁对谁的回复。通常会用 “A 回复 B:xxx”或者“张三:@李四 xxx”这两种形式。
换句话说,后端接口需要返回:评论、用户(评论的作者)、评论时间、对谁的回复(被评论人)。
你可能会想,一级评论和二级评论的字段好像不同啊,二级评论还多了个“谁对谁的回复”,果然,还是要拆开两张表。
其实不用,设计评论表时统一设计content、user_id、to_user_id、create_time字段即可,一级评论如果to_user_id用不上,可以空着(也可以认为一级评论是对文章作者的回复,或者干脆to_user_id字段设置为0或null)。
总之,同一张表在兼容多种类型时,应该以多的一方考虑。在表设计时,可以“多退”(用不上就空着呗),但不能“少补”(没有就真的没有了,除非修改表结构)。
这里给出较为可行的表设计:
SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;
-- ----------------------------
-- Table structure for t_comment
-- ----------------------------
DROP TABLE IF EXISTS `t_comment`;
CREATE TABLE `t_comment` (
`id` int(11) NOT NULL AUTO_INCREMENT COMMENT '评论id',
`pid` int(11) DEFAULT NULL COMMENT '所属一级评论的id,如果当前评论为一级,则为0',
`target_id` int(11) NOT NULL COMMENT '评论所属文章id',
`content` varchar(255) COLLATE utf8mb4_bin NOT NULL COMMENT '评论内容',
`user_id` varchar(255) COLLATE utf8mb4_bin NOT NULL COMMENT '该条评论的作者',
`to_user_id` varchar(255) COLLATE utf8mb4_bin DEFAULT NULL COMMENT '对谁回复,一级评论可以为null',
`likes_count` int(11) DEFAULT '0' COMMENT '当前评论的点赞数',
`create_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间',
`update_time` datetime NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
PRIMARY KEY (`id`),
KEY `key_target_id` (`target_id`) USING BTREE,
KEY `key_pid` (`pid`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_bin;
-- ----------------------------
-- Records of t_comment
-- ----------------------------
BEGIN;
INSERT INTO `t_comment` VALUES (1, 0, 10086, '这是第一条评论。', 'zhangsan', NULL, 1, '2020-03-17 11:06:39', '2020-03-17 14:06:13');
INSERT INTO `t_comment` VALUES (2, 0, 10086, '这是第二条评论。', 'lisi', NULL, 0, '2020-03-17 11:08:10', '2020-03-17 14:06:16');
INSERT INTO `t_comment` VALUES (3, 2, 10086, '你好啊,第二条评论。', 'zhangsan', 'lisi', 2, '2020-03-17 11:08:56', '2020-03-17 11:43:37');
INSERT INTO `t_comment` VALUES (4, 2, 10086, '哇,谢谢你的回复!', 'lisi', 'zhangsan', 0, '2020-03-17 11:09:57', '2020-03-17 12:02:40');
INSERT INTO `t_comment` VALUES (5, 0, 10086, '楼上两个细佬...', 'wangwu', NULL, 0, '2020-03-17 11:10:24', '2020-03-17 14:06:20');
INSERT INTO `t_comment` VALUES (6, 2, 10086, '回复一下而已,需要这么激动吗...', 'zhaoliu', 'lisi', 1, '2020-03-17 11:11:40', '2020-03-17 12:02:48');
COMMIT;
SET FOREIGN_KEY_CHECKS = 1;
为了方便展示时理清关系,这里我把user_id设置为VARCHAR,这样user_id就可以填入zhangsan、lisi,直观一些。
代码示例(通用Mapper)
Comment
@Data
@Table(name = "t_comment")
public class Comment {
/**
* 评论id
*/
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY, generator = "SELECT LAST_INSERT_ID()")
private Integer id;
/**
* 所属一级评论的id,如果当前评论为一级,则为0
*/
private Integer pid;
/**
* 评论所属文章id
*/
@Column(name = "target_id")
private Integer targetId;
/**
* 评论内容
*/
private String content;
/**
* 该条评论的作者
*/
@Column(name = "user_id")
private String userId;
/**
* 对谁回复,一级评论可以为null
*/
@Column(name = "to_user_id")
private String toUserId;
/**
* 当前评论的点赞数
*/
@Column(name = "likes_count")
private Integer likesCount;
/**
* 创建时间
*/
@Column(name = "create_time")
private Date createTime;
/**
* 更新时间
*/
@Column(name = "update_time")
private Date updateTime;
/**
* 该评论下的回复,非数据库字段,用 @Transient
*/
@Transient
private List<Comment> replies = new ArrayList<>();
}
CommentMapper
public interface CommentMapper extends Mapper<Comment> {
}
CommentService
@Service
public class CommentService {
@Autowired
private CommentMapper commentMapper;
public List<Comment> getAllCommentsByTargetId(Integer targetId) {
Example example = new Example(Comment.class);
Example.Criteria criteria = example.createCriteria();
criteria.andEqualTo("targetId", targetId);
example.setOrderByClause("id asc");
List<Comment> commentList = commentMapper.selectByExample(example);
return commentList;
}
}
启动类
@SpringBootApplication
@MapperScan("com.bravo")// 记得加扫描
public class SpringbootDemoApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootDemoApplication.class, args);
}
}
测试类
@RunWith(SpringRunner.class)
@SpringBootTest
public class CommentTest {
@Autowired
private CommentService commentService;
@Autowired
private ObjectMapper objectMapper;
@Test
public void testComment() throws JsonProcessingException {
// =========查出targetId下所有评论(一篇文章下的所有评论)==========
List<Comment> commentList = commentService.getAllCommentsByTargetId(10086);
// =========对平铺数据进行嵌套整理==========
// 最终结果
List<Comment> result = new ArrayList<>();
// list转map,建立索引
Map<Integer, Comment> commentMap = new HashMap<>();
for (Comment comment : commentList) {
commentMap.put(comment.getId(), comment);
}
// 嵌套数据
for (Comment comment : commentList) {
/**
* 归纳评论:对文章的评论是第一级,对文章的评论的评论是第二级,把第二级评论塞到对应的第一级评论下,作为replies
*
* 《静夜思》
* 床前明月光
* 疑似地上霜
* -----------------------
* a:第一级评论1
* a 回复 b:第二级评论1
* b 回复 a:第二级评论2
*
* c:第一级评论2
* c 回复 d:第二级评论3
* d 回复 c:第二级评论4
*/
if (comment.getPid() == 0) {
// 一级评论
result.add(comment);
} else{
// 二级评论,那么肯定有一级评论且firstComment一定不为null
Comment firstComment = commentMap.get(comment.getPid());
// 把二级评论塞到一级评论下
firstComment.getReplies().add(comment);
}
}
prettyPrint(result);
}
private void prettyPrint(List<Comment> commentList) throws JsonProcessingException {
System.out.println(objectMapper.writerWithDefaultPrettyPrinter().writeValueAsString(commentList));
}
}
结果展示
JSON返回值(顺序和层级结构已经处理好了,前端只要展示即可):
[
{
"id": 1,
"pid": 0,
"targetId": 10086,
"content": "这是第一条评论。",
"userId": "zhangsan",
"toUserId": null,
"likesCount": 1,
"createTime": "2020-03-17T03:06:39.000+0000",
"updateTime": "2020-03-17T06:06:13.000+0000",
"replies": []
},
{
"id": 2,
"pid": 0,
"targetId": 10086,
"content": "这是第二条评论。",
"userId": "lisi",
"toUserId": null,
"likesCount": 0,
"createTime": "2020-03-17T03:08:10.000+0000",
"updateTime": "2020-03-17T06:06:16.000+0000",
"replies": [
{
"id": 3,
"pid": 2,
"targetId": 10086,
"content": "你好啊,第二条评论。",
"userId": "zhangsan",
"toUserId": "lisi",
"likesCount": 2,
"createTime": "2020-03-17T03:08:56.000+0000",
"updateTime": "2020-03-17T03:43:37.000+0000",
"replies": []
},
{
"id": 4,
"pid": 2,
"targetId": 10086,
"content": "哇,谢谢你的回复!",
"userId": "lisi",
"toUserId": "zhangsan",
"likesCount": 0,
"createTime": "2020-03-17T03:09:57.000+0000",
"updateTime": "2020-03-17T04:02:40.000+0000",
"replies": []
},
{
"id": 6,
"pid": 2,
"targetId": 10086,
"content": "回复一下而已,需要这么激动吗...",
"userId": "zhaoliu",
"toUserId": "lisi",
"likesCount": 1,
"createTime": "2020-03-17T03:11:40.000+0000",
"updateTime": "2020-03-17T04:02:48.000+0000",
"replies": []
}
]
},
{
"id": 5,
"pid": 0,
"targetId": 10086,
"content": "楼上两个细佬...",
"userId": "wangwu",
"toUserId": null,
"likesCount": 0,
"createTime": "2020-03-17T03:10:24.000+0000",
"updateTime": "2020-03-17T06:06:20.000+0000",
"replies": []
}
]
扩展
之前提到了过,每条评论除了内容本身,还有用户信息:头像、昵称、简介等等:
而上面为了简单,JSON返回值中只有userId,并没有用户头像、昵称及个人简介。
解决办法也简单,一般评论肯定会做分页,所以一次查询的数量是有限的,我们可以查询出commentList后,用之前封装的ConvertUtil#resultToList收集所有评论的userId,再调用UserService.listUserInfoByIdList()查询所有用户信息,此时内存中有commentList和userList,而它们都有userId,不用我说大家也知道怎么做啦。
我们来看看掘金网站是怎么做的:
我们发现,把鼠标移到任意用户头像上时,会弹出一个tab页显示用户的信息,并且仔细观察的话,此时并没有触发异步请求,说明是后端嵌套好的。
[
{
"id": 2,
"pid": 0,
"targetId": 10086,
"content": "想咨询一个问题,就是如何获取跟 listview 一样某一个 item 的 view?",
"userId": "xiehan",
"toUserId": null,
"likesCount": 0,
"userInfo": {
"objectId": "57ab4807d342d30057867209",
"username": "谢函",
"avatarLarge": "",
"selfDescription": "",
"jobTitle": "",
"company": "",
"viewedEntriesCount": 423,
"collectedEntriesCount": 64,
"level": 0,
"isFollow": false
},
"toUserInfo": null,
"createTime": "2020-03-17T03:08:10.000+0000",
"updateTime": "2020-03-17T06:06:16.000+0000",
"replies": [
{
"id": 3,
"pid": 2,
"targetId": 10086,
"content": "RecyclerView.getChildAt",
"userId": "chengdazhi",
"toUserId": "xiehan",
"likesCount": 2,
"userInfo": {
"objectId": "56a9a4941532bc005304ab60",
"username": "程大治",
"avatarLarge": "https://user-gold-cdn.xitu.io/2016/11/29/f74e01b6a8cb2ced5da81e1aceac5e40",
"selfDescription": "计算机视觉研究 前Android开发",
"jobTitle": "科研实习",
"company": "微软亚洲研究院",
"viewedEntriesCount": 497,
"collectedEntriesCount": 41,
"level": 0,
"isFollow": false
},
"toUserInfo": {
"objectId": "57ab4807d342d30057867209",
"username": "谢函",
"avatarLarge": "",
"selfDescription": "",
"jobTitle": "",
"company": "",
"viewedEntriesCount": 423,
"collectedEntriesCount": 64,
"level": 0,
"isFollow": false
},
"createTime": "2020-03-17T03:08:56.000+0000",
"updateTime": "2020-03-17T03:43:37.000+0000",
"replies": []
}
]
}
]
当鼠标触发hover事件时,直接从当前Comment中取出UserInfo展示。
当然,掘金的这种做法会使得同一个作者的多条评论中带有相同的UserInfo,前后端传递到数据有很大的冗余。
如果是你,会怎么改进呢?
作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO
进群,大家一起学习,一起进步,一起对抗互联网寒冬