首頁 > 軟體

Java單表實現評論回覆功能(多種實現方式)

2023-03-05 14:02:46

1.簡介

最近在寫畢業設計的時候發現需要實現一個評論功能,然後看了一下掘金和csdn的評論區,如何實現評論功能?

評論功能有多種實現方式:

  • 單層型
  • 套娃型(多層型)
  • 兩層型

單層型:

套娃型:

兩層型:

2.功能實現圖

3.資料庫設計

這個地方有個answer_id 很容易讓人迷糊:是回覆哪個使用者的id

CREATE TABLE `tb_blog_comments`  (
  `id` bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '主鍵',
  `user_id` bigint(20) UNSIGNED NOT NULL COMMENT '使用者id',
  `blog_id` bigint(20) UNSIGNED NOT NULL COMMENT '探店id',
  `parent_id` bigint(20) UNSIGNED NOT NULL COMMENT '關聯的1級評論id,如果是一級評論,則值為0',
  `answer_id` bigint(20) UNSIGNED NOT NULL COMMENT '回覆的評論id',
  `content` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT '回覆的內容',
  `liked` int(8) UNSIGNED NULL DEFAULT 0 COMMENT '點贊數',
  `status` tinyint(1) UNSIGNED NULL DEFAULT 0 COMMENT '狀態,0:正常,1:被舉報,2:禁止檢視',
  `create_time` timestamp(0) NOT NULL DEFAULT CURRENT_TIMESTAMP(0) COMMENT '建立時間',
  `update_time` timestamp(0) NOT NULL DEFAULT CURRENT_TIMESTAMP(0) ON UPDATE CURRENT_TIMESTAMP(0) COMMENT '更新時間',
  PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = COMPACT;

SET FOREIGN_KEY_CHECKS = 1;

4.實體類

import com.baomidou.mybatisplus.annotation.IdType;
import com.baomidou.mybatisplus.annotation.TableField;
import com.baomidou.mybatisplus.annotation.TableId;
import com.baomidou.mybatisplus.annotation.TableName;
import com.fasterxml.jackson.annotation.JsonFormat;
import lombok.Data;
import lombok.EqualsAndHashCode;
import lombok.experimental.Accessors;

import java.io.Serializable;
import java.time.LocalDateTime;
import java.util.List;

/**
 * <p>
 * 
 * </p>
 *
 * @author 尹穩健
 * @since 2022-11-09
 */
@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@TableName("tb_blog_comments")
public class BlogComments implements Serializable {

    private static final long serialVersionUID = 1L;

    /**
     * 主鍵
     */
    @TableId(value = "id", type = IdType.AUTO)
    private Long id;

    /**
     * 使用者id
     */
    private Long userId;

    /**
     * 探店id
     */
    private Long blogId;

    /**
     * 關聯的1級評論id,如果是一級評論,則值為0
     */
    private Long parentId;

    /**
     * 回覆的評論id
     */
    private Long answerId;

    /**
     * 回覆的內容
     */
    private String content;

    /**
     * 點贊數
     */
    private Integer liked;

    /**
     * 狀態,0:正常,1:被舉報,2:禁止檢視
     */
    private Boolean status;

    /**
     * 建立時間
     */
    @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")
    private LocalDateTime createTime;

    /**
     * 更新時間
     */
    private LocalDateTime updateTime;

    /**
     * 是否點贊
     */
    @TableField(exist = false)
    private Boolean isLike;

    /**
     * 子評論
     */
    @TableField(exist = false)
    List<BlogComments> children;

    /**
     * 評論者的暱稱
     */
    @TableField(exist = false)
    private String nickName;

    /** 評論者的頭像 */
    @TableField(exist = false)
    private String icon;

    /** 評論者的上級暱稱 */
    @TableField(exist = false)
    private String pNickName;

    /** 評論者的的上級頭像 */
    @TableField(exist = false)
    private String pIcon;
}

5.實現思路

  • 因為有評論區有兩層,所以肯定有一個parent_id,這樣你才能知道你是哪個評論下面的回覆內容,如果繼續評論,那麼那條評論的parent_id還是之前那條評論的parent_id,而不是那條子評論的id。
  • 回覆內容也同樣是一個評論實體類,只不過是一個集合,所以用List 儲存,泛型使用實體類
  • 我的功能實現也用到了父評論的使用者名稱和頭像,這樣可以更好看出這是誰評論的,回覆給誰的評論

6.功能實現

6.1 Sql入手

首先,我們需要知道自己需要哪些資料,返回給前端

如果你連mybatis都不會,那就看下思路吧
從這裡獲取到了評論表的所有資料,以及評論的人的資訊,比如說暱稱、頭像、id,可以展示在前端

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" >
<mapper namespace="com.sky.mapper.BlogCommentsMapper">
    <select id="findCommentDetail" resultType="com.sky.pojo.BlogComments">
        SELECT
            bl.*,
            u.icon,
            u.nick_name
        FROM `tb_blog_comments` bl
        left join tb_user u
        on u.id = bl.user_id
        where bl.blog_id = #{blogId}
        order by bl.id desc
    </select>
</mapper>

對應的mapper介面

package com.sky.mapper;

import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.sky.pojo.BlogComments;
import org.apache.ibatis.annotations.Mapper;

import java.util.List;

/**
 * <p>
 *  Mapper 介面
 * </p>
 *
 * @author 尹穩健
 * @since 2022-11-09
 */
@Mapper
public interface BlogCommentsMapper extends BaseMapper<BlogComments> {

    /**
     * 查詢所有的評論資訊
     * @param blogId
     * @return
     */
    List<BlogComments> findCommentDetail(Long blogId);
}

6.2 業務實現

  • 1.首先我們需要從資料中獲取所有的資料
  • 2.然後我們需要找到所有的一級評論,一級評論就是最高階,他不在誰的下面,他就是最大的,我這裡在新增評論的時候前端做了處理,只要是一級評論,他的paren_id = 0
  • 3.然後我們需要從一級評論下面新增他下面所有的子評論
  • 最主要的就是如何將父級評論的資訊新增到自己的資料中?
  • 4.通過子評論中的paren_id 找到父評論,然後通過子評論的answer_id == 父評論的user_id 這樣就可以拿到父評論的那一條資料
  • 最後通過Optional 新增到子評論的資料中
@Override
public Result showBlogComments(Long blogId) {
    // 先將資料庫中的資料全部查詢出來,包括評論作者的資訊,暱稱和頭像
    List<BlogComments> blogComments = blogCommentsMapper.findCommentDetail(blogId);
    // 獲取所有的一級評論
    List<BlogComments> rootComments = blogComments.stream().filter(blogComments1 -> blogComments1.getParentId() == 0).collect(Collectors.toList());
    // 從一級評論中獲取回覆評論
    for (BlogComments rootComment : rootComments) {
        // 回覆的評論
        List<BlogComments> comments = blogComments.stream()
                .filter(blogComment -> blogComment.getParentId().equals(rootComment.getId()))
                .collect(Collectors.toList());
        // 回覆評論中含有父級評論的資訊
        comments.forEach(comment -> {
            // 無法判斷pComment是否存在,可以使用Optional
            Optional<BlogComments> pComment
                    = blogComments
                    .stream()
                    // 獲取所有的評論的回覆id也就是父級id的userid,這樣就可以獲取到父級評論的資訊
                    .filter(blogComment -> comment.getAnswerId().equals(blogComment.getUserId())).findFirst();
            // 這裡使用了Optional 只有pcomment!=null 的時候才會執行下面的程式碼
            pComment.ifPresent(v -> {
                comment.setPNickName(v.getNickName());
                comment.setPIcon(v.getIcon());
            });
            // 判斷是否點贊
            isBlogCommentLiked(comment);
        });
        rootComment.setChildren(comments);
        // 判斷是否點贊
        isBlogCommentLiked(rootComment);
    }
    return Result.ok(rootComments);
}


7.前端實現

因為前端程式碼很多,只copy關鍵程式碼

<html>
<body>
<div class="comment-list">
          <div class="comment-box" style="display: block" v-for="comment in commnetList" :key="comment.id">
            <div style="display:flex">
              <!-- 評論者頭像 -->
              <div class="comment-icon">
                <img :src="comment.icon" alt="">
              </div>
              <!-- 評論div -->
              <div class="comment-info">
                <!-- 評論者暱稱 -->
                <div class="comment-user">
                  {{comment.nickName}}
                </div>
                <!-- 評論時間 -->
                <div style="display: flex">
                  {{comment.createTime}}
                  <!-- 評論點贊,回覆按鈕 -->
                  <div style="margin-left: 42%;display: flex;">
                    <div @click="addCommnetLike(comment)" style="display: flex">
                      <svg t="1646634642977" class="icon" viewBox="0 0 1024 1024" version="1.1"
                        xmlns="http://www.w3.org/2000/svg" p-id="2187" width="14" height="14">
                        <path
                          d="M160 944c0 8.8-7.2 16-16 16h-32c-26.5 0-48-21.5-48-48V528c0-26.5 21.5-48 48-48h32c8.8 0 16 7.2 16 16v448zM96 416c-53 0-96 43-96 96v416c0 53 43 96 96 96h96c17.7 0 32-14.3 32-32V448c0-17.7-14.3-32-32-32H96zM505.6 64c16.2 0 26.4 8.7 31 13.9 4.6 5.2 12.1 16.3 10.3 32.4l-23.5 203.4c-4.9 42.2 8.6 84.6 36.8 116.4 28.3 31.7 68.9 49.9 111.4 49.9h271.2c6.6 0 10.8 3.3 13.2 6.1s5 7.5 4 14l-48 303.4c-6.9 43.6-29.1 83.4-62.7 112C815.8 944.2 773 960 728.9 960h-317c-33.1 0-59.9-26.8-59.9-59.9v-455c0-6.1 1.7-12 5-17.1 69.5-109 106.4-234.2 107-364h41.6z m0-64h-44.9C427.2 0 400 27.2 400 60.7c0 127.1-39.1 251.2-112 355.3v484.1c0 68.4 55.5 123.9 123.9 123.9h317c122.7 0 227.2-89.3 246.3-210.5l47.9-303.4c7.8-49.4-30.4-94.1-80.4-94.1H671.6c-50.9 0-90.5-44.4-84.6-95l23.5-203.4C617.7 55 568.7 0 505.6 0z"
                          p-id="2188" :fill="comment.isLike ? '#ff6633' : '#82848a'"></path>
                      </svg>
                      &nbsp;{{comment.liked}}
                    </div>
                    <!--  評論回覆 -->
                    <div style=" display:flex">
                      &nbsp;&nbsp;&nbsp;&nbsp;
                      <el-dropdown trigger="click" size="mini" placement="top" type="mini">
                        <i class="el-icon-more"></i>
                        <el-dropdown-menu>
                          <el-dropdown-item>
                            <div @click="replyCommentForm(comment)">
                              回覆
                            </div>
                          </el-dropdown-item>
                          <el-dropdown-item>
                            <div v-if="comment.userId == user.id" @click="deleteComment(comment.id)">
                              刪除
                            </div>
                          </el-dropdown-item>
                        </el-dropdown-menu>
                      </el-dropdown>
                      &nbsp;
                    </div>
                  </div>
                </div>
                <!-- 評論主題 : 評論內容,點贊,回覆 -->
                <div style="padding: 5px 0; font-size: 14px;display: flex;">
                  <!-- 評論內容 -->
                  <div>
                    {{comment.content}}
                  </div>
                </div>
              </div>
            </div>

            <!-- 回覆的內容 -->
            <div v-if="comment.children.length" style="padding-left: 5px;">
              <div v-for="reply in comment.children" :key="reply.id" style="padding: 5px 10px">
                <div style="display: flex">
                  <!-- 評論者頭像 -->
                  <div class="comment-icon">
                    <img :src="reply.icon" alt="">
                  </div>
                  <!-- 評論div -->
                  <div class="comment-info">
                    <!-- 評論者暱稱 -->
                    <div class="comment-user">
                      {{reply.nickName}} &nbsp;回覆: {{reply.pnickName}}
                    </div>
                    <!-- 評論時間 -->
                    <div style="display: flex">
                      {{reply.createTime}}
                      <!-- 評論點贊,回覆按鈕 -->
                      <div style="margin-left: 40%;display: flex;">
                        <div style="display: flex" @click="addCommnetLike(reply)">
                          <svg t="1646634642977" class="icon" viewBox="0 0 1024 1024" version="1.1"
                            xmlns="http://www.w3.org/2000/svg" p-id="2187" width="14" height="14">
                            <path
                              d="M160 944c0 8.8-7.2 16-16 16h-32c-26.5 0-48-21.5-48-48V528c0-26.5 21.5-48 48-48h32c8.8 0 16 7.2 16 16v448zM96 416c-53 0-96 43-96 96v416c0 53 43 96 96 96h96c17.7 0 32-14.3 32-32V448c0-17.7-14.3-32-32-32H96zM505.6 64c16.2 0 26.4 8.7 31 13.9 4.6 5.2 12.1 16.3 10.3 32.4l-23.5 203.4c-4.9 42.2 8.6 84.6 36.8 116.4 28.3 31.7 68.9 49.9 111.4 49.9h271.2c6.6 0 10.8 3.3 13.2 6.1s5 7.5 4 14l-48 303.4c-6.9 43.6-29.1 83.4-62.7 112C815.8 944.2 773 960 728.9 960h-317c-33.1 0-59.9-26.8-59.9-59.9v-455c0-6.1 1.7-12 5-17.1 69.5-109 106.4-234.2 107-364h41.6z m0-64h-44.9C427.2 0 400 27.2 400 60.7c0 127.1-39.1 251.2-112 355.3v484.1c0 68.4 55.5 123.9 123.9 123.9h317c122.7 0 227.2-89.3 246.3-210.5l47.9-303.4c7.8-49.4-30.4-94.1-80.4-94.1H671.6c-50.9 0-90.5-44.4-84.6-95l23.5-203.4C617.7 55 568.7 0 505.6 0z"
                              p-id="2188" :fill="reply.isLike ? '#ff6633' : '#82848a'"></path>
                          </svg>
                          &nbsp;{{reply.liked}}
                        </div>
                        <div style="display:flex">
                          &nbsp;&nbsp;&nbsp;&nbsp;
                          <el-dropdown trigger="click" size="mini" placement="top" type="mini">
                            <i class="el-icon-more"></i>
                            <el-dropdown-menu>
                              <el-dropdown-item>
                                <div @click="replyCommentForm(reply)">
                                  回覆
                                </div>
                              </el-dropdown-item>
                              <el-dropdown-item>
                                <div v-if="reply.userId == user.id" @click="deleteComment(reply.id)">
                                  刪除
                                </div>
                              </el-dropdown-item>
                            </el-dropdown-menu>
                          </el-dropdown>
                          &nbsp;
                        </div>
                      </div>
                    </div>
                    <!-- 評論內容 -->
                    <div style="padding: 5px 0; font-size: 14px;display: flex;">
                      <!-- 評論內容 -->
                      <div>
                        {{reply.content}}
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </div>
          </div>
        </div>
<script>
    let each = function (ary, callback) {
      for (let i = 0, l = ary.length; i < l; i++) {
        if (callback(ary[i], i) === false) break
      }
    }
    const app = new Vue({
      el: "#app",
      data: {
        util,
        showPopover: false,
        actions: [{ text: '回覆', icon: '' }, { text: '刪除', icon: '' }],
        commentsTotal: '',  // 評論總數
        checkCommentInputVisible: false,  // 判斷是否點選回覆
        commnetList: {},  // 評論回覆列表
        placeholderText: '發條友善評論吧~~',
        commentForm: {
          content: '',
          userId: '',
          blogId: '',
          parentId: 0,
          answerId: 0,
        },   // 評論表單
        blog: {},
        shop: {},
        likes: [],
        user: {}, // 登入使用者
      methods: {

        replyCommentForm(comment) {
          this.placeholderText = "回覆 " + comment.nickName
          this.checkCommentInputVisible = true;
          if (comment.parentId == 0) {
            this.commentForm.parentId = comment.id;
          } else {
            this.commentForm.parentId = comment.parentId;
          }
          this.commentForm.answerId = comment.userId;
        },
        submitCommentForm() {
          this.commentForm.userId = this.user.id;
          this.commentForm.blogId = this.blog.id;
          axios.post("/blog-comments/saveBlogComment", this.commentForm)
            .then(res => {
              this.loadComments(this.blog.id);
              this.checkCommentInputVisible = false;
            })
            .catch(err => this.$message.error(err))
        },
        loadComments(id) {
          axios.get("/blog-comments/showBlogComments/" + id)
            .then(res => {
              this.commnetList = res.data
            })
            .catch(err => this.$message.err(error))
        },
        }
  </script>
  </body>

</html>

8.最終成果

點贊、評論、回覆、刪除

到此這篇關於Java單表實現評論回覆功能(多種實現方式)的文章就介紹到這了,更多相關Java評論回覆內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com