二级评论列表是很常见的功能,文章记录了新手用Java实现的具体逻辑。
整体实现逻辑是先用2个sql,分别查出两层数据。然后用java在service中实现数据组装,返给前端。这种实现思路好处是SQL简洁,逻辑分明,便于维护。
一:需求场景
一级评论的列表,平铺展示。当涉及多人回复,或者两个人多次对话后, 留言逻辑看着非常混乱。如下图
当改造为二级列表后,数据展示更加直观。如下图, 演示地址:CodingLife
二、SQL实现
先用2个SQL分别查出两层数据,先查parent_id为空的数据,就是第一层数据。
再用MyBatis的嵌套查询,配置in语句查出所有的二级数据。
# 一级评论<select id="getuserSideLevel1" resultMap="indexResultMap">select*fromleave_message mlleft join user useron user.id = ml.leave_idwhereparent_id is nullorder bycreate_time desclimit #{start},#{size}</select># 二级评论<select id="getuserSideLevel2" resultMap="indexResultMap">select*fromleave_message mlleft join user useron user.id = ml.leave_idwhereparent_id in<foreach collection="parentIds" item="parentId" open="(" separator="," close=")">#{parentId}</foreach>order bycreate_time asc</select>
三、Java组装
使用Java将两级数据组装在一起,下面为service的具体代码。
// 先查分页范围内的一级数据List<LeaveMessage> level1List = leaveMessageMapper.getuserSideLevel1(start,size);// 取出一级数据的idList<Integer> parentIdArray = new ArrayList<>();for (int i = 0; i < level1List.size(); i++) {LeaveMessage level1Item = level1List.get(i);Integer id = level1Item.getId(); // 获取 idparentIdArray.add(id);}// 再查以上一级数据的二级数据List<LeaveMessage> level2List = leaveMessageMapper.getuserSideLevel2(parentIdArray);// 组装如上两级数据for (int i = 0; i < level1List.size(); i++) {LeaveMessage level1Item = level1List.get(i);List<LeaveMessage> childList = new ArrayList<>();for (int j = 0; j < level2List.size(); j++) {LeaveMessage child = level2List.get(j);if (level1Item.getId().equals(child.getParentId())) {childList.add(child);}}level1Item.setChild(childList);}return ApiResponse.success(level1List);
四、源码
演示地址为:CodingLife,源码地址为:Git