From a1b2b40b6829661541e366d6c5a763d73aa220cf Mon Sep 17 00:00:00 2001 From: Veal98 <1912420914@qq.com> Date: Thu, 11 Feb 2021 21:51:04 +0800 Subject: [PATCH] Add Gitee Pages --- README.md | 2 +- docs/.vuepress/config.js | 1 + docs/Guide.md | 2 +- .../core/.temp/app-enhancers/global-components-4.js | 8 ++++++++ .../core/.temp/app-enhancers/global-components-5.js | 8 ++++++++ .../core/.temp/app-enhancers/global-components-6.js | 8 ++++++++ .../core/.temp/app-enhancers/global-components-7.js | 8 ++++++++ .../@vuepress/core/.temp/internal/app-enhancers.js | 2 +- node_modules/@vuepress/core/.temp/internal/siteData.js | 8 +++++--- .../.cache/vuepress/9ed3d496a895af1692a95b1d6395b8fc.json | 2 +- .../.cache/vuepress/b4c93b53937ab1eb6dd25d96729e6256.json | 2 +- .../.cache/vuepress/cf95a4f56065996d75717bef77e96a78.json | 2 +- 12 files changed, 44 insertions(+), 9 deletions(-) create mode 100644 node_modules/@vuepress/core/.temp/app-enhancers/global-components-4.js create mode 100644 node_modules/@vuepress/core/.temp/app-enhancers/global-components-5.js create mode 100644 node_modules/@vuepress/core/.temp/app-enhancers/global-components-6.js create mode 100644 node_modules/@vuepress/core/.temp/app-enhancers/global-components-7.js diff --git a/README.md b/README.md index 3c834bad..91aa4675 100644 --- a/README.md +++ b/README.md @@ -4,7 +4,7 @@ ## 📚 从本项目你能学到什么 -- 学会主流的 Java Web 开发技术和框架(Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch 等) +- 学会主流的 Java Web 开发技术和框架(Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch、Spring Security 等) - 了解一个真实的 Web 项目从开发到部署的整个流程(本项目配套有大量图例和详细教程,以帮助小伙伴快速上手) - 掌握本项目中涉及的核心技术点以及常见面试题和解析 diff --git a/docs/.vuepress/config.js b/docs/.vuepress/config.js index 50192fcd..c4ef7c7d 100644 --- a/docs/.vuepress/config.js +++ b/docs/.vuepress/config.js @@ -1,6 +1,7 @@ module.exports = { title: '开源社区系统 — Echo', // 设置网站标题 description : '一款基于 SpringBoot + MyBatis + MySQL + Redis + Kafka + Elasticsearch + ... 实现的开源社区系统,并提供详细的开发文档和配套教程', + base : '/Echo/', themeConfig : { nav : [ { diff --git a/docs/Guide.md b/docs/Guide.md index 3c834bad..91aa4675 100644 --- a/docs/Guide.md +++ b/docs/Guide.md @@ -4,7 +4,7 @@ ## 📚 从本项目你能学到什么 -- 学会主流的 Java Web 开发技术和框架(Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch 等) +- 学会主流的 Java Web 开发技术和框架(Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch、Spring Security 等) - 了解一个真实的 Web 项目从开发到部署的整个流程(本项目配套有大量图例和详细教程,以帮助小伙伴快速上手) - 掌握本项目中涉及的核心技术点以及常见面试题和解析 diff --git a/node_modules/@vuepress/core/.temp/app-enhancers/global-components-4.js b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-4.js new file mode 100644 index 00000000..11c72262 --- /dev/null +++ b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-4.js @@ -0,0 +1,8 @@ +import Vue from 'vue' + +Vue.component("Badge", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\Badge")) +Vue.component("CodeBlock", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeBlock")) +Vue.component("CodeGroup", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeGroup")) + + +export default {} \ No newline at end of file diff --git a/node_modules/@vuepress/core/.temp/app-enhancers/global-components-5.js b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-5.js new file mode 100644 index 00000000..e522bb53 --- /dev/null +++ b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-5.js @@ -0,0 +1,8 @@ +import Vue from 'vue' + +Vue.component("CodeBlock", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeBlock")) +Vue.component("Badge", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\Badge")) +Vue.component("CodeGroup", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeGroup")) + + +export default {} \ No newline at end of file diff --git a/node_modules/@vuepress/core/.temp/app-enhancers/global-components-6.js b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-6.js new file mode 100644 index 00000000..11c72262 --- /dev/null +++ b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-6.js @@ -0,0 +1,8 @@ +import Vue from 'vue' + +Vue.component("Badge", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\Badge")) +Vue.component("CodeBlock", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeBlock")) +Vue.component("CodeGroup", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeGroup")) + + +export default {} \ No newline at end of file diff --git a/node_modules/@vuepress/core/.temp/app-enhancers/global-components-7.js b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-7.js new file mode 100644 index 00000000..11c72262 --- /dev/null +++ b/node_modules/@vuepress/core/.temp/app-enhancers/global-components-7.js @@ -0,0 +1,8 @@ +import Vue from 'vue' + +Vue.component("Badge", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\Badge")) +Vue.component("CodeBlock", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeBlock")) +Vue.component("CodeGroup", () => import("E:\\GreateCommunity\\node_modules\\@vuepress\\theme-default\\global-components\\CodeGroup")) + + +export default {} \ No newline at end of file diff --git a/node_modules/@vuepress/core/.temp/internal/app-enhancers.js b/node_modules/@vuepress/core/.temp/internal/app-enhancers.js index a7e2f95c..a902e91d 100644 --- a/node_modules/@vuepress/core/.temp/internal/app-enhancers.js +++ b/node_modules/@vuepress/core/.temp/internal/app-enhancers.js @@ -1,6 +1,6 @@ import m0 from "E:\\GreateCommunity\\node_modules\\@vuepress\\core\\.temp\\app-enhancers\\0.js" import m1 from "E:\\GreateCommunity\\node_modules\\@vuepress\\core\\.temp\\app-enhancers\\data-block.js" -import m2 from "E:\\GreateCommunity\\node_modules\\@vuepress\\core\\.temp\\app-enhancers\\global-components-3.js" +import m2 from "E:\\GreateCommunity\\node_modules\\@vuepress\\core\\.temp\\app-enhancers\\global-components-7.js" import m3 from "E:\\GreateCommunity\\node_modules\\@vuepress\\core\\.temp\\app-enhancers\\1.js" export default [ diff --git a/node_modules/@vuepress/core/.temp/internal/siteData.js b/node_modules/@vuepress/core/.temp/internal/siteData.js index dcce90d8..d3c9cc72 100644 --- a/node_modules/@vuepress/core/.temp/internal/siteData.js +++ b/node_modules/@vuepress/core/.temp/internal/siteData.js @@ -4,7 +4,7 @@ export const siteData = { "title": "开源社区系统 — Echo", "description": "一款基于 SpringBoot + MyBatis + MySQL + Redis + Kafka + Elasticsearch + ... 实现的开源社区系统,并提供详细的开发文档和配套教程", - "base": "/", + "base": "/Echo/", "headTags": [], "pages": [ { @@ -35,7 +35,8 @@ export const siteData = { "regularPath": "/", "relativePath": "README.md", "key": "v-269b69fd", - "path": "/" + "path": "/", + "lastUpdated": "2/11/2021, 9:31:41 PM" }, { "title": "Echo — 开源社区系统", @@ -210,7 +211,8 @@ export const siteData = { "title": "👏 鸣谢", "slug": "👏-鸣谢" } - ] + ], + "lastUpdated": "2/11/2021, 9:31:41 PM" } ], "themeConfig": { diff --git a/node_modules/@vuepress/core/node_modules/.cache/vuepress/9ed3d496a895af1692a95b1d6395b8fc.json b/node_modules/@vuepress/core/node_modules/.cache/vuepress/9ed3d496a895af1692a95b1d6395b8fc.json index c69a10f7..aeb971d9 100644 --- a/node_modules/@vuepress/core/node_modules/.cache/vuepress/9ed3d496a895af1692a95b1d6395b8fc.json +++ b/node_modules/@vuepress/core/node_modules/.cache/vuepress/9ed3d496a895af1692a95b1d6395b8fc.json @@ -1 +1 @@ -{"remainingRequest":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\loaders\\templateLoader.js??vue-loader-options!E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js??ref--1-0!E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js??ref--1-1!E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js??ref--1-2!E:\\GreateCommunity\\docs\\Guide.md?vue&type=template&id=1a70d707&","dependencies":[{"path":"E:\\GreateCommunity\\docs\\Guide.md","mtime":1613049784185},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\loaders\\templateLoader.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:var render = function() {
  var _vm = this
  var _h = _vm.$createElement
  var _c = _vm._self._c || _h
  return _c(
    "ContentSlotsDistributor",
    { attrs: { "slot-key": _vm.$parent.slotKey } },
    [
      _c("h1", { attrs: { id: "echo-开源社区系统" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#echo-开源社区系统" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" Echo — 开源社区系统")
      ]),
      _vm._v(" "),
      _c("hr"),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📚-从本项目你能学到什么" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#📚-从本项目你能学到什么" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 📚 从本项目你能学到什么")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v(
            "学会主流的 Java Web 开发技术和框架（Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch 等）"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "了解一个真实的 Web 项目从开发到部署的整个流程（本项目配套有大量图例和详细教程，以帮助小伙伴快速上手）"
          )
        ]),
        _vm._v(" "),
        _c("li", [_vm._v("掌握本项目中涉及的核心技术点以及常见面试题和解析")])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🏄‍-在线体验与文档地址" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#🏄‍-在线体验与文档地址" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 🏄‍ 在线体验与文档地址")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v(
            "在线体验：项目已经部署到腾讯云服务器，各位小伙伴们可直接线上体验："
          ),
          _c(
            "a",
            {
              attrs: {
                href: "http://1.15.127.74/",
                target: "_blank",
                rel: "noopener noreferrer"
              }
            },
            [_vm._v("http://1.15.127.74/"), _c("OutboundLink")],
            1
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "文档地址：文档通过 Vuepress + Gitee Pages 生成，在线访问地址："
          )
        ])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "💻-核心技术栈" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#💻-核心技术栈" } },
          [_vm._v("#")]
        ),
        _vm._v(" 💻 核心技术栈")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("后端：")]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("Spring")]),
        _vm._v(" "),
        _c("li", [_vm._v("Spring Boot 2.1.5 RELEASE")]),
        _vm._v(" "),
        _c("li", [_vm._v("Spring MVC")]),
        _vm._v(" "),
        _c("li", [_vm._v("ORM：MyBatis")]),
        _vm._v(" "),
        _c("li", [_vm._v("数据库：MySQL 5.7")]),
        _vm._v(" "),
        _c("li", [_vm._v("分布式缓存：Redis")]),
        _vm._v(" "),
        _c("li", [_vm._v("本地缓存：Caffeine")]),
        _vm._v(" "),
        _c("li", [_vm._v("消息队列：Kafka 2.13-2.7.0")]),
        _vm._v(" "),
        _c("li", [_vm._v("搜索引擎：Elasticsearch 6.4.3")]),
        _vm._v(" "),
        _c("li", [_vm._v("安全：Spring Security")]),
        _vm._v(" "),
        _c("li", [_vm._v("邮件：Spring Mail")]),
        _vm._v(" "),
        _c("li", [_vm._v("分布式定时任务：Spring Quartz")]),
        _vm._v(" "),
        _c("li", [_vm._v("日志：SLF4J（日志接口） + Logback（日志实现）")])
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("前端：")]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("Thymeleaf")]),
        _vm._v(" "),
        _c("li", [_vm._v("Bootstrap 4.x")]),
        _vm._v(" "),
        _c("li", [_vm._v("Jquery")]),
        _vm._v(" "),
        _c("li", [_vm._v("Ajax")])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🔨-开发环境" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🔨-开发环境" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🔨 开发环境")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("操作系统：Windows 10")]),
        _vm._v(" "),
        _c("li", [_vm._v("构建工具：Apache Maven")]),
        _vm._v(" "),
        _c("li", [_vm._v("集成开发工具：Intellij IDEA")]),
        _vm._v(" "),
        _c("li", [_vm._v("应用服务器：Apache Tomcat")]),
        _vm._v(" "),
        _c("li", [_vm._v("接口测试工具：Postman")]),
        _vm._v(" "),
        _c("li", [_vm._v("压力测试工具：Apache JMeter")]),
        _vm._v(" "),
        _c("li", [_vm._v("版本控制工具：Git")]),
        _vm._v(" "),
        _c("li", [_vm._v("Java 版本：8")])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🎀-界面展示" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🎀-界面展示" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🎀 界面展示")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("首页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205641.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("登录页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205558.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("帖子详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205741.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("个人主页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205820.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("朋友私信页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205857.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("私信详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205948.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("系统通知页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210122.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("通知详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210152.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("账号设置页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210238.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("数据统计页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210323.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("搜索详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210531.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🎨-功能列表" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🎨-功能列表" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🎨 功能列表")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208222403.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("注册")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v(
                "用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活"
              )
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v(
                "向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）"
              )
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("登录 | 登出")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _c("p", [
                _vm._v(
                  "进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）"
                )
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v(
                  "用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis"
                )
              ]),
              _vm._v(" "),
              _c("p", [
                _vm._v(
                  "注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）"
                )
              ])
            ]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("勾选记住我，则延长登录凭证有效时间")])]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v("用户登录成功，将用户信息短暂存入 Redis（1 小时）")
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v(
                  "用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息"
                )
              ])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("账号设置")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("修改头像\n"),
              _c("ul", [
                _c("li", [_vm._v("将用户选择的头像图片文件上传至七牛云服务器")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("修改密码")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("帖子模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("发布帖子（过滤敏感词），将其存入 MySQL")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("分页显示所有的帖子\n"),
              _c("ul", [
                _c("li", [_vm._v("支持按照 “发帖时间” 显示")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持按照 “热度排行” 显示（Spring Quartz）")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("查看帖子详情")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security + Thymeleaf Security）\n"),
              _c("ul", [
                _c("li", [_vm._v("未登录用户无法发帖")]),
                _vm._v(" "),
                _c("li", [
                  _vm._v("“版主” 可以看到帖子的置顶和加精按钮并执行相应操作")
                ]),
                _vm._v(" "),
                _c("li", [
                  _vm._v("“管理员” 可以看到帖子的删除按钮并执行相应操作")
                ]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "“普通用户” 无法看到帖子的置顶、加精、删除按钮，也无法执行相应操作"
                  )
                ])
              ])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("评论模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("发布对帖子的评论（过滤敏感词），将其存入 MySQL")
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("分页显示评论")]),
            _vm._v(" "),
            _c("li", [_vm._v("发布对评论的回复（过滤敏感词）")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用评论功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("私信模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("发送私信（过滤敏感词）")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("私信列表\n"),
              _c("ul", [
                _c("li", [_vm._v("查询当前用户的会话列表")]),
                _vm._v(" "),
                _c("li", [_vm._v("每个会话只显示一条最新的私信")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持分页显示")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("私信详情\n"),
              _c("ul", [
                _c("li", [_vm._v("查询某个会话所包含的所有私信")]),
                _vm._v(" "),
                _c("li", [_vm._v("访问私信详情时，将显示的私信设为已读状态")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持分页显示")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用私信功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v("[x] "),
            _c("strong", [_vm._v("统一处理 404 / 500 异常")])
          ]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("普通请求异常")]),
            _vm._v(" "),
            _c("li", [_vm._v("异步请求异常")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("统一记录日志")])])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("点赞模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_c("p", [_vm._v("支持对帖子、评论/回复点赞")])]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("第 1 次点赞，第 2 次取消点赞")])]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("首页统计帖子的点赞数量")])]),
            _vm._v(" "),
            _c("li", [
              _c("p", [_vm._v("详情页统计帖子和评论/回复的点赞数量")])
            ]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v("详情页显示当前登录用户的点赞状态（赞过了则显示已赞）")
              ])
            ]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("统计我的获赞数量")])]),
            _vm._v(" "),
            _c("li", [
              _c("p", [_vm._v("权限管理（Spring Security）")]),
              _vm._v(" "),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用点赞相关功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("关注模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("关注功能")]),
            _vm._v(" "),
            _c("li", [_vm._v("取消关注功能")]),
            _vm._v(" "),
            _c("li", [_vm._v("统计用户的关注数和粉丝数")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("我的关注列表（查询某个用户关注的人），支持分页")
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("我的粉丝列表（查询某个用户的粉丝），支持分页")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用关注相关功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("系统通知模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("通知列表\n"),
              _c("ul", [
                _c("li", [_vm._v("显示评论、点赞、关注三种类型的通知")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("通知详情\n"),
              _c("ul", [
                _c("li", [_vm._v("分页显示某一类主题所包含的通知")]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "进入某种类型的系统通知详情，则将该页的所有未读的系统通知状态设置为已读"
                  )
                ])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("未读数量\n"),
              _c("ul", [
                _c("li", [_vm._v("分别显示每种类型的系统通知的未读数量")]),
                _vm._v(" "),
                _c("li", [_vm._v("显示所有系统通知的未读数量")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("导航栏显示所有消息的未读数量（未读私信 + 未读系统通知）")
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用系统通知功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("搜索模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("发布事件\n"),
              _c("ul", [
                _c("li", [
                  _vm._v(
                    "发布帖子时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器"
                  )
                ]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "为帖子增加评论时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器"
                  )
                ])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("搜索服务\n"),
              _c("ul", [
                _c("li", [_vm._v("从 Elasticsearch 服务器搜索帖子")]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "从 Elasticsearch 服务器删除帖子（当帖子从数据库中被删除时）"
                  )
                ])
              ])
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("显示搜索结果")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v("[x] "),
            _c("strong", [_vm._v("网站数据统计")]),
            _vm._v("（管理员专属）")
          ]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("独立访客 UV\n"),
              _c("ul", [
                _c("li", [_vm._v("存入 Redis 的 HyperLogLog")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持单日查询和区间日期查询")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("日活跃用户 DAU\n"),
              _c("ul", [
                _c("li", [_vm._v("存入 Redis 的 Bitmap")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持单日查询和区间日期查询")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("只有管理员可以查看网站数据统计")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] 优化网站性能")]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("使用本地缓存 Caffeine 缓存热帖列表以及所有用户帖子的总数")
            ])
          ])
        ])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🔐-待实现及优化" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🔐-待实现及优化" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🔐 待实现及优化")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "以下是我个人发现的本项目存在的问题，但是暂时没有头绪无法解决，集思广益，欢迎各位小伙伴提 PR 解决："
        )
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v("[ ] 注册模块无法正常跳转到操作提示界面（本地运行没有问题）")
        ]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 评论功能的前端显示部分存在 Bug")]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 查询我的评论（未完善）")])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "以下是我觉得本项目还可以添加的功能，同样欢迎各位小伙伴提 issue 指出还可以增加哪些功能，或者直接提 PR 实现该功能："
        )
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("[ ] 忘记密码（发送邮件找回密码）")]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 查询我的点赞")]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 管理员对帖子的二次点击取消置顶功能")]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "[ ] 管理员对已删除帖子的恢复功能（本项目中的删除帖子并未将其从数据库中删除，只是将其状态设置为了拉黑）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🌱-本地运行" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🌱-本地运行" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🌱 本地运行")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("各位如果需要将项目部署在本地进行测试，以下环境请提前备好：")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("Java 8")]),
        _vm._v(" "),
        _c("li", [_vm._v("MySQL 5.7")]),
        _vm._v(" "),
        _c("li", [_vm._v("Redis")]),
        _vm._v(" "),
        _c("li", [_vm._v("Kafka 2.13-2.7.0")]),
        _vm._v(" "),
        _c("li", [_vm._v("Elasticsearch 6.4.3")])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("然后"),
        _c("strong", [
          _vm._v("修改配置文件中的信息为你自己的本地环境，直接运行是运行不了的")
        ]),
        _vm._v("，而且相关私密信息我全部用 xxxxxxx 代替了。")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("本地运行需要修改的配置文件信息如下：")]),
      _vm._v(" "),
      _c("p", [
        _vm._v("1）"),
        _c("code", [_vm._v("application-develop.properties")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("MySQL")]),
        _vm._v(" "),
        _c("li", [_vm._v("Spring Mail（邮箱需要开启 SMTP 服务）")]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "Kafka：consumer.group-id（该字段见 Kafka 安装包中的 consumer.proerties，可自行修改, 修改完毕后需要重启 Kafka）"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "Elasticsearch：cluster-name（该字段见 Elasticsearch 安装包中的 elasticsearch.yml，可自行修改）"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "七牛云（需要新建一个七牛云的对象存储空间，用来存放上传的头像图片）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("2）"),
        _c("code", [_vm._v("logback-spring-develop.xml")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("ul", [_c("li", [_vm._v("LOG_PATH：日志存放的位置")])]),
      _vm._v(" "),
      _c("p", [_vm._v("每次运行需要打开：")]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("MySQL")]),
        _vm._v(" "),
        _c("li", [_vm._v("Redis")]),
        _vm._v(" "),
        _c("li", [_vm._v("Elasticsearch")]),
        _vm._v(" "),
        _c("li", [_vm._v("Kafka")])
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("另外，还需要事件建好数据库表，详细见下文。")]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📜-数据库设计" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#📜-数据库设计" } },
          [_vm._v("#")]
        ),
        _vm._v(" 📜 数据库设计")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("用户 "), _c("code", [_vm._v("user")]), _vm._v("：")]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DROP")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("IF")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("EXISTS")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("user")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("SET")
            ]),
            _vm._v(" character_set_client "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" utf8mb4 "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("user")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("username"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("50")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("password"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("50")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("salt"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("50")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("email"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("100")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("type")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-普通用户; 1-超级管理员; 2-版主;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-未激活; 1-已激活;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("activation_code"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("100")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("header_url"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("200")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_username"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("username"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("20")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_email"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("email"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("20")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("101")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [
        _vm._v("讨论帖 "),
        _c("code", [_vm._v("discuss_post")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DROP")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("IF")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("EXISTS")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("discuss_post"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("SET")
            ]),
            _vm._v(" character_set_client "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" utf8mb4 "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("discuss_post"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("title"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("100")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("content"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("text")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("type")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-普通; 1-置顶;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-正常; 1-精华; 2-拉黑;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("comment_count"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("score"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("double")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [
        _vm._v("评论（回复）"),
        _c("code", [_vm._v("comment")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("comment")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("entity_type"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'评论目标的类别：1 帖子；2 评论 '")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("entity_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'评论目标的 id'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("target_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'指明对谁进行评论'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("content"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("text")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'状态：0 正常；1 禁用'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_entity_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("entity_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("247")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [_vm._v("私信 "), _c("code", [_vm._v("message")]), _vm._v("：")]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DROP")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("IF")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("EXISTS")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("message"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("SET")
            ]),
            _vm._v(" character_set_client "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" utf8mb4 "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("message"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("from_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("to_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("conversation_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("45")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("content"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("text")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-未读;1-已读;2-删除;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_from_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("from_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_to_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("to_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_conversation_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("conversation_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("h2", { attrs: { id: "🌌-理想的部署架构" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#🌌-理想的部署架构" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 🌌 理想的部署架构")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("我每个都只部署了一台，以下是理想的部署架构：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211204207.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🎯-功能逻辑图" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🎯-功能逻辑图" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🎯 功能逻辑图")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("画了一些不是那么严谨的图帮助各位小伙伴理清思绪。")]),
      _vm._v(" "),
      _c("blockquote", [
        _c("p", [_vm._v("单向绿色箭头：")]),
        _vm._v(" "),
        _c("ul", [
          _c("li", [
            _vm._v(
              "前端模板 -> Controller：表示这个前端模板中有一个超链接是由这个 Controller 处理的"
            )
          ]),
          _vm._v(" "),
          _c("li", [
            _vm._v(
              "Controller -> 前端模板：表示这个 Controller 会像该前端模板传递数据或者跳转"
            )
          ])
        ]),
        _vm._v(" "),
        _c("p", [
          _vm._v(
            "双向绿色箭头：表示 Controller 和前端模板之间进行参数的相互传递或使用"
          )
        ]),
        _vm._v(" "),
        _c("p", [_vm._v("单向蓝色箭头： A -> B，表示 A 方法调用了 B 方法")]),
        _vm._v(" "),
        _c("p", [_vm._v("单向红色箭头：数据库或缓存操作")])
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "注册" } }, [
        _c("a", { staticClass: "header-anchor", attrs: { href: "#注册" } }, [
          _vm._v("#")
        ]),
        _vm._v(" 注册")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v("用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活")
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v("向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）")
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210204222249.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "登录-登出" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#登录-登出" } },
          [_vm._v("#")]
        ),
        _vm._v(" 登录 | 登出")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _c("p", [
            _vm._v(
              "进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）"
            )
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v(
              "用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis"
            )
          ]),
          _vm._v(" "),
          _c("p", [
            _vm._v(
              "注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）"
            )
          ])
        ]),
        _vm._v(" "),
        _c("li", [_c("p", [_vm._v("勾选记住我，则延长登录凭证有效时间")])]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("用户登录成功，将用户信息短暂存入 Redis（1 小时）")])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v(
              "用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息"
            )
          ])
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "下图是登录模块的功能逻辑图，并没有使用 Spring Security 提供的认证逻辑（我觉得这个模块是最复杂的，这张图其实很多细节还没有画全）"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210204233233.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "分页显示所有的帖子" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#分页显示所有的帖子" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 分页显示所有的帖子")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("支持按照 “发帖时间” 显示")]),
        _vm._v(" "),
        _c("li", [_vm._v("支持按照 “热度排行” 显示（Spring Quartz）")]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "将热帖列表和所有帖子的总数存入本地缓存 Caffeine（利用分布式定时任务 Spring Quartz 每隔一段时间就刷新计算帖子的热度/分数 — 见下文，而 Caffeine 里的数据更新不用我们操心，它天生就会自动的更新它拥有的数据，给它一个初始化方法就完事儿）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210204222822.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "账号设置" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#账号设置" } },
          [_vm._v("#")]
        ),
        _vm._v(" 账号设置")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v("修改头像（异步请求）\n"),
          _c("ul", [
            _c("li", [_vm._v("将用户选择的头像图片文件上传至七牛云服务器")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [_vm._v("修改密码")])
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("此处只画出修改头像：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210206121201.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "发布帖子-异步请求" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#发布帖子-异步请求" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 发布帖子（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210206122521.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "显示评论及相关信息" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#显示评论及相关信息" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 显示评论及相关信息")
      ]),
      _vm._v(" "),
      _c("blockquote", [
        _c("p", [
          _vm._v(
            "评论部分前端的名称显示有些缺陷，有兴趣的小伙伴欢迎提 PR 解决~"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "关于评论模块需要注意的就是评论表的设计，把握其中字段的含义，才能透彻了解这个功能的逻辑。"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "评论 Comment 的目标类型（帖子，评论） entityType 和 entityId 以及对哪个用户进行评论/回复 targetId 是由前端传递给 DiscussPostController 的"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207150925.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("一个帖子的详情页需要封装的信息大概如下：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207151328.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "添加评论-事务管理" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#添加评论-事务管理" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 添加评论（事务管理）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207122908.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "私信列表和详情页" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#私信列表和详情页" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 私信列表和详情页")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207161130.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "发送私信-异步请求" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#发送私信-异步请求" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 发送私信（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207161500.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "点赞-异步请求" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#点赞-异步请求" } },
          [_vm._v("#")]
        ),
        _vm._v(" 点赞（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "将点赞相关信息存入 Redis 的数据结构 set 中。其中，key 命名为  "
        ),
        _c("code", [_vm._v("like:entity:entityType:entityId")]),
        _vm._v("，value 即点赞用户的 id。比如 key =  "),
        _c("code", [_vm._v("like:entity:2:246")]),
        _vm._v("  value =  "),
        _c("code", [_vm._v("11")]),
        _vm._v(" 表示用户 11 对实体类型 2 即评论进行了点赞，该评论的 id 是 246")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("某个用户的获赞数量对应的存储在 Redis 中的 key 是 "),
        _c("code", [_vm._v("like:user:userId")]),
        _vm._v("，value 就是这个用户的获赞数量")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207165837.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "我的获赞数量" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#我的获赞数量" } },
          [_vm._v("#")]
        ),
        _vm._v(" 我的获赞数量")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207170003.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "关注-异步请求" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#关注-异步请求" } },
          [_vm._v("#")]
        ),
        _vm._v(" 关注（异步请求）")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v(
            "若 A 关注了 B，则 A 是 B 的粉丝 Follower，B 是 A 的目标 Followee"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "关注的目标可以是用户、帖子、题目等，在实现时将这些目标抽象为实体（目前只做了关注用户）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "将某个用户关注的实体相关信息存储在 Redis 的数据结构 zset 中：key 是 "
        ),
        _c("code", [_vm._v("followee:userId:entityType")]),
        _vm._v(" ，对应的 value 是 "),
        _c("code", [_vm._v("zset(entityId, now)")]),
        _vm._v(" ，以关注的时间进行排序。比如说 "),
        _c("code", [_vm._v("followee:111:3")]),
        _vm._v(" 对应的value "),
        _c("code", [_vm._v("(20, 2020-02-03-xxxx)")]),
        _vm._v(
          "，表明用户 111 关注了实体类型为 3 即人(用户)，该帖子的 id 是 20，关注该帖子的时间是 2020-02-03-xxxx"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "同样的，将某个实体拥有的粉丝相关信息也存储在 Redis 的数据结构 zset 中：key 是 "
        ),
        _c("code", [_vm._v("follower:entityType:entityId")]),
        _vm._v("，对应的 value 是 "),
        _c("code", [_vm._v("zset(userId, now)")]),
        _vm._v("，以关注的时间进行排序")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207174046.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "关注列表" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#关注列表" } },
          [_vm._v("#")]
        ),
        _vm._v(" 关注列表")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207175621.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "发送系统通知" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#发送系统通知" } },
          [_vm._v("#")]
        ),
        _vm._v(" 发送系统通知")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207182917.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "显示系统通知" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#显示系统通知" } },
          [_vm._v("#")]
        ),
        _vm._v(" 显示系统通知")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208153059.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "搜索" } }, [
        _c("a", { staticClass: "header-anchor", attrs: { href: "#搜索" } }, [
          _vm._v("#")
        ]),
        _vm._v(" 搜索")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208161936.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("类似的，置顶、加精也会触发发帖事件，就不再图里面画出来了。")
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "置顶加精删除-异步请求" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#置顶加精删除-异步请求" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 置顶加精删除（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208171729.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "网站数据统计" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#网站数据统计" } },
          [_vm._v("#")]
        ),
        _vm._v(" 网站数据统计")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208170801.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "帖子热度计算" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#帖子热度计算" } },
          [_vm._v("#")]
        ),
        _vm._v(" 帖子热度计算")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "每次发生点赞（给帖子点赞）、评论（给帖子评论）、加精的时候，就将这些帖子信息存入缓存 Redis 中，然后通过分布式的定时任务 Spring Quartz，每隔一段时间就从缓存中取出这些帖子进行计算分数。"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("帖子分数/热度计算公式：分数（热度） = 权重 + 发帖距离天数")
      ]),
      _vm._v(" "),
      _c("div", { staticClass: "language-java extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-java" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token comment" } }, [
              _vm._v("// 计算权重")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("double")
            ]),
            _vm._v(" w "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("wonderful "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("?")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("75")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v(":")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("0")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("+")
            ]),
            _vm._v(" commentCount "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("10")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("+")
            ]),
            _vm._v(" likeCount "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("2")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token comment" } }, [
              _vm._v("// 分数 = 权重 + 发帖距离天数")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("double")
            ]),
            _vm._v(" score "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token class-name" } }, [
              _vm._v("Math")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("log10")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token class-name" } }, [
              _vm._v("Math")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("max")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("w"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("1")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n        "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("+")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("post"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("getCreateTime")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("getTime")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("-")
            ]),
            _vm._v(" epoch"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("getTime")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("/")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("1000")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("3600")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("24")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208173636.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📖-配套教程" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#📖-配套教程" } },
          [_vm._v("#")]
        ),
        _vm._v(" 📖 配套教程")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "想要自己从零开始实现这个项目或者深入理解的小伙伴，可以扫描下方二维码关注公众号『"
        ),
        _c("strong", [_vm._v("飞天小牛肉")]),
        _vm._v(
          "』，第一时间获取配套教程, 不仅会详细解释本项目涉及的各大技术点，还会汇总相关的常见面试题，目前尚在更新中。"
        )
      ]),
      _vm._v(" "),
      _c("img", {
        staticStyle: { zoom: "67%" },
        attrs: {
          src:
            "https://gitee.com/veal98/images/raw/master/img/20210204145531.png"
        }
      }),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📞-联系我" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#📞-联系我" } },
          [_vm._v("#")]
        ),
        _vm._v(" 📞 联系我")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("有什么问题也可以添加我的微信，记得备注来意：格式 "),
        _c("u", [_vm._v("（学校或公司 - 姓名或昵称 - 来意）")])
      ]),
      _vm._v(" "),
      _c("img", {
        attrs: {
          width: "260px",
          src:
            "https://gitee.com/veal98/images/raw/master/img/微信图片_20210105121328.jpg"
        }
      }),
      _vm._v(" "),
      _c("h2", { attrs: { id: "👏-鸣谢" } }, [
        _c("a", { staticClass: "header-anchor", attrs: { href: "#👏-鸣谢" } }, [
          _vm._v("#")
        ]),
        _vm._v(" 👏 鸣谢")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("本项目参考"),
        _c(
          "a",
          {
            attrs: {
              href: "https://www.nowcoder.com/",
              target: "_blank",
              rel: "noopener noreferrer"
            }
          },
          [_vm._v("牛客网"), _c("OutboundLink")],
          1
        ),
        _vm._v(" — Java 高级工程师课程，感谢老师和平台")
      ])
    ]
  )
}
var staticRenderFns = []
render._withStripped = true

export { render, staticRenderFns }"}]} \ No newline at end of file +{"remainingRequest":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\loaders\\templateLoader.js??vue-loader-options!E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js??ref--1-0!E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js??ref--1-1!E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js??ref--1-2!E:\\GreateCommunity\\docs\\Guide.md?vue&type=template&id=1a70d707&","dependencies":[{"path":"E:\\GreateCommunity\\docs\\Guide.md","mtime":1613050811891},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\loaders\\templateLoader.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:var render = function() {
  var _vm = this
  var _h = _vm.$createElement
  var _c = _vm._self._c || _h
  return _c(
    "ContentSlotsDistributor",
    { attrs: { "slot-key": _vm.$parent.slotKey } },
    [
      _c("h1", { attrs: { id: "echo-开源社区系统" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#echo-开源社区系统" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" Echo — 开源社区系统")
      ]),
      _vm._v(" "),
      _c("hr"),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📚-从本项目你能学到什么" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#📚-从本项目你能学到什么" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 📚 从本项目你能学到什么")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v(
            "学会主流的 Java Web 开发技术和框架（Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch、Spring Security 等）"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "了解一个真实的 Web 项目从开发到部署的整个流程（本项目配套有大量图例和详细教程，以帮助小伙伴快速上手）"
          )
        ]),
        _vm._v(" "),
        _c("li", [_vm._v("掌握本项目中涉及的核心技术点以及常见面试题和解析")])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🏄‍-在线体验与文档地址" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#🏄‍-在线体验与文档地址" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 🏄‍ 在线体验与文档地址")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v(
            "在线体验：项目已经部署到腾讯云服务器，各位小伙伴们可直接线上体验："
          ),
          _c(
            "a",
            {
              attrs: {
                href: "http://1.15.127.74/",
                target: "_blank",
                rel: "noopener noreferrer"
              }
            },
            [_vm._v("http://1.15.127.74/"), _c("OutboundLink")],
            1
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "文档地址：文档通过 Vuepress + Gitee Pages 生成，在线访问地址："
          )
        ])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "💻-核心技术栈" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#💻-核心技术栈" } },
          [_vm._v("#")]
        ),
        _vm._v(" 💻 核心技术栈")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("后端：")]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("Spring")]),
        _vm._v(" "),
        _c("li", [_vm._v("Spring Boot 2.1.5 RELEASE")]),
        _vm._v(" "),
        _c("li", [_vm._v("Spring MVC")]),
        _vm._v(" "),
        _c("li", [_vm._v("ORM：MyBatis")]),
        _vm._v(" "),
        _c("li", [_vm._v("数据库：MySQL 5.7")]),
        _vm._v(" "),
        _c("li", [_vm._v("分布式缓存：Redis")]),
        _vm._v(" "),
        _c("li", [_vm._v("本地缓存：Caffeine")]),
        _vm._v(" "),
        _c("li", [_vm._v("消息队列：Kafka 2.13-2.7.0")]),
        _vm._v(" "),
        _c("li", [_vm._v("搜索引擎：Elasticsearch 6.4.3")]),
        _vm._v(" "),
        _c("li", [_vm._v("安全：Spring Security")]),
        _vm._v(" "),
        _c("li", [_vm._v("邮件：Spring Mail")]),
        _vm._v(" "),
        _c("li", [_vm._v("分布式定时任务：Spring Quartz")]),
        _vm._v(" "),
        _c("li", [_vm._v("日志：SLF4J（日志接口） + Logback（日志实现）")])
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("前端：")]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("Thymeleaf")]),
        _vm._v(" "),
        _c("li", [_vm._v("Bootstrap 4.x")]),
        _vm._v(" "),
        _c("li", [_vm._v("Jquery")]),
        _vm._v(" "),
        _c("li", [_vm._v("Ajax")])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🔨-开发环境" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🔨-开发环境" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🔨 开发环境")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("操作系统：Windows 10")]),
        _vm._v(" "),
        _c("li", [_vm._v("构建工具：Apache Maven")]),
        _vm._v(" "),
        _c("li", [_vm._v("集成开发工具：Intellij IDEA")]),
        _vm._v(" "),
        _c("li", [_vm._v("应用服务器：Apache Tomcat")]),
        _vm._v(" "),
        _c("li", [_vm._v("接口测试工具：Postman")]),
        _vm._v(" "),
        _c("li", [_vm._v("压力测试工具：Apache JMeter")]),
        _vm._v(" "),
        _c("li", [_vm._v("版本控制工具：Git")]),
        _vm._v(" "),
        _c("li", [_vm._v("Java 版本：8")])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🎀-界面展示" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🎀-界面展示" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🎀 界面展示")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("首页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205641.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("登录页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205558.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("帖子详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205741.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("个人主页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205820.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("朋友私信页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205857.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("私信详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211205948.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("系统通知页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210122.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("通知详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210152.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("账号设置页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210238.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("数据统计页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210323.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("搜索详情页：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211210531.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🎨-功能列表" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🎨-功能列表" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🎨 功能列表")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208222403.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("注册")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v(
                "用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活"
              )
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v(
                "向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）"
              )
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("登录 | 登出")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _c("p", [
                _vm._v(
                  "进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）"
                )
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v(
                  "用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis"
                )
              ]),
              _vm._v(" "),
              _c("p", [
                _vm._v(
                  "注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）"
                )
              ])
            ]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("勾选记住我，则延长登录凭证有效时间")])]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v("用户登录成功，将用户信息短暂存入 Redis（1 小时）")
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v(
                  "用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息"
                )
              ])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("账号设置")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("修改头像\n"),
              _c("ul", [
                _c("li", [_vm._v("将用户选择的头像图片文件上传至七牛云服务器")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("修改密码")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("帖子模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("发布帖子（过滤敏感词），将其存入 MySQL")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("分页显示所有的帖子\n"),
              _c("ul", [
                _c("li", [_vm._v("支持按照 “发帖时间” 显示")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持按照 “热度排行” 显示（Spring Quartz）")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("查看帖子详情")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security + Thymeleaf Security）\n"),
              _c("ul", [
                _c("li", [_vm._v("未登录用户无法发帖")]),
                _vm._v(" "),
                _c("li", [
                  _vm._v("“版主” 可以看到帖子的置顶和加精按钮并执行相应操作")
                ]),
                _vm._v(" "),
                _c("li", [
                  _vm._v("“管理员” 可以看到帖子的删除按钮并执行相应操作")
                ]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "“普通用户” 无法看到帖子的置顶、加精、删除按钮，也无法执行相应操作"
                  )
                ])
              ])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("评论模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("发布对帖子的评论（过滤敏感词），将其存入 MySQL")
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("分页显示评论")]),
            _vm._v(" "),
            _c("li", [_vm._v("发布对评论的回复（过滤敏感词）")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用评论功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("私信模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("发送私信（过滤敏感词）")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("私信列表\n"),
              _c("ul", [
                _c("li", [_vm._v("查询当前用户的会话列表")]),
                _vm._v(" "),
                _c("li", [_vm._v("每个会话只显示一条最新的私信")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持分页显示")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("私信详情\n"),
              _c("ul", [
                _c("li", [_vm._v("查询某个会话所包含的所有私信")]),
                _vm._v(" "),
                _c("li", [_vm._v("访问私信详情时，将显示的私信设为已读状态")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持分页显示")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用私信功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v("[x] "),
            _c("strong", [_vm._v("统一处理 404 / 500 异常")])
          ]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("普通请求异常")]),
            _vm._v(" "),
            _c("li", [_vm._v("异步请求异常")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("统一记录日志")])])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("点赞模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_c("p", [_vm._v("支持对帖子、评论/回复点赞")])]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("第 1 次点赞，第 2 次取消点赞")])]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("首页统计帖子的点赞数量")])]),
            _vm._v(" "),
            _c("li", [
              _c("p", [_vm._v("详情页统计帖子和评论/回复的点赞数量")])
            ]),
            _vm._v(" "),
            _c("li", [
              _c("p", [
                _vm._v("详情页显示当前登录用户的点赞状态（赞过了则显示已赞）")
              ])
            ]),
            _vm._v(" "),
            _c("li", [_c("p", [_vm._v("统计我的获赞数量")])]),
            _vm._v(" "),
            _c("li", [
              _c("p", [_vm._v("权限管理（Spring Security）")]),
              _vm._v(" "),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用点赞相关功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("关注模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [_vm._v("关注功能")]),
            _vm._v(" "),
            _c("li", [_vm._v("取消关注功能")]),
            _vm._v(" "),
            _c("li", [_vm._v("统计用户的关注数和粉丝数")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("我的关注列表（查询某个用户关注的人），支持分页")
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("我的粉丝列表（查询某个用户的粉丝），支持分页")]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用关注相关功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("系统通知模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("通知列表\n"),
              _c("ul", [
                _c("li", [_vm._v("显示评论、点赞、关注三种类型的通知")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("通知详情\n"),
              _c("ul", [
                _c("li", [_vm._v("分页显示某一类主题所包含的通知")]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "进入某种类型的系统通知详情，则将该页的所有未读的系统通知状态设置为已读"
                  )
                ])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("未读数量\n"),
              _c("ul", [
                _c("li", [_vm._v("分别显示每种类型的系统通知的未读数量")]),
                _vm._v(" "),
                _c("li", [_vm._v("显示所有系统通知的未读数量")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("导航栏显示所有消息的未读数量（未读私信 + 未读系统通知）")
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("未登录用户无法使用系统通知功能")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] "), _c("strong", [_vm._v("搜索模块")])]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("发布事件\n"),
              _c("ul", [
                _c("li", [
                  _vm._v(
                    "发布帖子时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器"
                  )
                ]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "为帖子增加评论时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器"
                  )
                ])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("搜索服务\n"),
              _c("ul", [
                _c("li", [_vm._v("从 Elasticsearch 服务器搜索帖子")]),
                _vm._v(" "),
                _c("li", [
                  _vm._v(
                    "从 Elasticsearch 服务器删除帖子（当帖子从数据库中被删除时）"
                  )
                ])
              ])
            ]),
            _vm._v(" "),
            _c("li", [_vm._v("显示搜索结果")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v("[x] "),
            _c("strong", [_vm._v("网站数据统计")]),
            _vm._v("（管理员专属）")
          ]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("独立访客 UV\n"),
              _c("ul", [
                _c("li", [_vm._v("存入 Redis 的 HyperLogLog")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持单日查询和区间日期查询")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("日活跃用户 DAU\n"),
              _c("ul", [
                _c("li", [_vm._v("存入 Redis 的 Bitmap")]),
                _vm._v(" "),
                _c("li", [_vm._v("支持单日查询和区间日期查询")])
              ])
            ]),
            _vm._v(" "),
            _c("li", [
              _vm._v("权限管理（Spring Security）\n"),
              _c("ul", [_c("li", [_vm._v("只有管理员可以查看网站数据统计")])])
            ])
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("[x] 优化网站性能")]),
          _vm._v(" "),
          _c("ul", [
            _c("li", [
              _vm._v("使用本地缓存 Caffeine 缓存热帖列表以及所有用户帖子的总数")
            ])
          ])
        ])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🔐-待实现及优化" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🔐-待实现及优化" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🔐 待实现及优化")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "以下是我个人发现的本项目存在的问题，但是暂时没有头绪无法解决，集思广益，欢迎各位小伙伴提 PR 解决："
        )
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v("[ ] 注册模块无法正常跳转到操作提示界面（本地运行没有问题）")
        ]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 评论功能的前端显示部分存在 Bug")]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 查询我的评论（未完善）")])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "以下是我觉得本项目还可以添加的功能，同样欢迎各位小伙伴提 issue 指出还可以增加哪些功能，或者直接提 PR 实现该功能："
        )
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("[ ] 忘记密码（发送邮件找回密码）")]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 查询我的点赞")]),
        _vm._v(" "),
        _c("li", [_vm._v("[ ] 管理员对帖子的二次点击取消置顶功能")]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "[ ] 管理员对已删除帖子的恢复功能（本项目中的删除帖子并未将其从数据库中删除，只是将其状态设置为了拉黑）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🌱-本地运行" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🌱-本地运行" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🌱 本地运行")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("各位如果需要将项目部署在本地进行测试，以下环境请提前备好：")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("Java 8")]),
        _vm._v(" "),
        _c("li", [_vm._v("MySQL 5.7")]),
        _vm._v(" "),
        _c("li", [_vm._v("Redis")]),
        _vm._v(" "),
        _c("li", [_vm._v("Kafka 2.13-2.7.0")]),
        _vm._v(" "),
        _c("li", [_vm._v("Elasticsearch 6.4.3")])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("然后"),
        _c("strong", [
          _vm._v("修改配置文件中的信息为你自己的本地环境，直接运行是运行不了的")
        ]),
        _vm._v("，而且相关私密信息我全部用 xxxxxxx 代替了。")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("本地运行需要修改的配置文件信息如下：")]),
      _vm._v(" "),
      _c("p", [
        _vm._v("1）"),
        _c("code", [_vm._v("application-develop.properties")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("MySQL")]),
        _vm._v(" "),
        _c("li", [_vm._v("Spring Mail（邮箱需要开启 SMTP 服务）")]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "Kafka：consumer.group-id（该字段见 Kafka 安装包中的 consumer.proerties，可自行修改, 修改完毕后需要重启 Kafka）"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "Elasticsearch：cluster-name（该字段见 Elasticsearch 安装包中的 elasticsearch.yml，可自行修改）"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "七牛云（需要新建一个七牛云的对象存储空间，用来存放上传的头像图片）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("2）"),
        _c("code", [_vm._v("logback-spring-develop.xml")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("ul", [_c("li", [_vm._v("LOG_PATH：日志存放的位置")])]),
      _vm._v(" "),
      _c("p", [_vm._v("每次运行需要打开：")]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("MySQL")]),
        _vm._v(" "),
        _c("li", [_vm._v("Redis")]),
        _vm._v(" "),
        _c("li", [_vm._v("Elasticsearch")]),
        _vm._v(" "),
        _c("li", [_vm._v("Kafka")])
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("另外，还需要事件建好数据库表，详细见下文。")]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📜-数据库设计" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#📜-数据库设计" } },
          [_vm._v("#")]
        ),
        _vm._v(" 📜 数据库设计")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("用户 "), _c("code", [_vm._v("user")]), _vm._v("：")]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DROP")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("IF")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("EXISTS")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("user")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("SET")
            ]),
            _vm._v(" character_set_client "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" utf8mb4 "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("user")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("username"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("50")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("password"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("50")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("salt"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("50")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("email"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("100")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("type")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-普通用户; 1-超级管理员; 2-版主;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-未激活; 1-已激活;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("activation_code"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("100")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("header_url"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("200")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_username"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("username"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("20")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_email"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("email"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("20")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("101")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [
        _vm._v("讨论帖 "),
        _c("code", [_vm._v("discuss_post")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DROP")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("IF")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("EXISTS")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("discuss_post"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("SET")
            ]),
            _vm._v(" character_set_client "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" utf8mb4 "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("discuss_post"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("title"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("100")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("content"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("text")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("type")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-普通; 1-置顶;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-正常; 1-精华; 2-拉黑;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("comment_count"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("score"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("double")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [
        _vm._v("评论（回复）"),
        _c("code", [_vm._v("comment")]),
        _vm._v("：")
      ]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("comment")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("entity_type"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'评论目标的类别：1 帖子；2 评论 '")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("entity_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'评论目标的 id'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("target_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'指明对谁进行评论'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("content"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("text")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'状态：0 正常；1 禁用'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("user_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_entity_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("entity_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("247")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [_vm._v("私信 "), _c("code", [_vm._v("message")]), _vm._v("：")]),
      _vm._v(" "),
      _c("div", { staticClass: "language-sql extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-sql" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DROP")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("IF")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("EXISTS")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("message"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("SET")
            ]),
            _vm._v(" character_set_client "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" utf8mb4 "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CREATE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("TABLE")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("message"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("AUTO_INCREMENT")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("from_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("to_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("conversation_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("varchar")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("45")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("NOT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("content"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("text")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("status")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("int")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("11")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("COMMENT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token string" } }, [
              _vm._v("'0-未读;1-已读;2-删除;'")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("create_time"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("timestamp")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token boolean" } }, [
              _vm._v("NULL")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("PRIMARY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_from_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("from_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_to_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("to_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v("\n  "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("KEY")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("index_conversation_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _vm._v("conversation_id"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("`")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("ENGINE")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("InnoDB")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("DEFAULT")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("CHARSET")
            ]),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v("utf8"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("h2", { attrs: { id: "🌌-理想的部署架构" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#🌌-理想的部署架构" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 🌌 理想的部署架构")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("我每个都只部署了一台，以下是理想的部署架构：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210211204207.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "🎯-功能逻辑图" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#🎯-功能逻辑图" } },
          [_vm._v("#")]
        ),
        _vm._v(" 🎯 功能逻辑图")
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("画了一些不是那么严谨的图帮助各位小伙伴理清思绪。")]),
      _vm._v(" "),
      _c("blockquote", [
        _c("p", [_vm._v("单向绿色箭头：")]),
        _vm._v(" "),
        _c("ul", [
          _c("li", [
            _vm._v(
              "前端模板 -> Controller：表示这个前端模板中有一个超链接是由这个 Controller 处理的"
            )
          ]),
          _vm._v(" "),
          _c("li", [
            _vm._v(
              "Controller -> 前端模板：表示这个 Controller 会像该前端模板传递数据或者跳转"
            )
          ])
        ]),
        _vm._v(" "),
        _c("p", [
          _vm._v(
            "双向绿色箭头：表示 Controller 和前端模板之间进行参数的相互传递或使用"
          )
        ]),
        _vm._v(" "),
        _c("p", [_vm._v("单向蓝色箭头： A -> B，表示 A 方法调用了 B 方法")]),
        _vm._v(" "),
        _c("p", [_vm._v("单向红色箭头：数据库或缓存操作")])
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "注册" } }, [
        _c("a", { staticClass: "header-anchor", attrs: { href: "#注册" } }, [
          _vm._v("#")
        ]),
        _vm._v(" 注册")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v("用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活")
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v("向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）")
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210204222249.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "登录-登出" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#登录-登出" } },
          [_vm._v("#")]
        ),
        _vm._v(" 登录 | 登出")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _c("p", [
            _vm._v(
              "进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）"
            )
          ])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v(
              "用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis"
            )
          ]),
          _vm._v(" "),
          _c("p", [
            _vm._v(
              "注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）"
            )
          ])
        ]),
        _vm._v(" "),
        _c("li", [_c("p", [_vm._v("勾选记住我，则延长登录凭证有效时间")])]),
        _vm._v(" "),
        _c("li", [
          _c("p", [_vm._v("用户登录成功，将用户信息短暂存入 Redis（1 小时）")])
        ]),
        _vm._v(" "),
        _c("li", [
          _c("p", [
            _vm._v(
              "用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息"
            )
          ])
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "下图是登录模块的功能逻辑图，并没有使用 Spring Security 提供的认证逻辑（我觉得这个模块是最复杂的，这张图其实很多细节还没有画全）"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210204233233.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "分页显示所有的帖子" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#分页显示所有的帖子" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 分页显示所有的帖子")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [_vm._v("支持按照 “发帖时间” 显示")]),
        _vm._v(" "),
        _c("li", [_vm._v("支持按照 “热度排行” 显示（Spring Quartz）")]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "将热帖列表和所有帖子的总数存入本地缓存 Caffeine（利用分布式定时任务 Spring Quartz 每隔一段时间就刷新计算帖子的热度/分数 — 见下文，而 Caffeine 里的数据更新不用我们操心，它天生就会自动的更新它拥有的数据，给它一个初始化方法就完事儿）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210204222822.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "账号设置" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#账号设置" } },
          [_vm._v("#")]
        ),
        _vm._v(" 账号设置")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v("修改头像（异步请求）\n"),
          _c("ul", [
            _c("li", [_vm._v("将用户选择的头像图片文件上传至七牛云服务器")])
          ])
        ]),
        _vm._v(" "),
        _c("li", [_vm._v("修改密码")])
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("此处只画出修改头像：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210206121201.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "发布帖子-异步请求" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#发布帖子-异步请求" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 发布帖子（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210206122521.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "显示评论及相关信息" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#显示评论及相关信息" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 显示评论及相关信息")
      ]),
      _vm._v(" "),
      _c("blockquote", [
        _c("p", [
          _vm._v(
            "评论部分前端的名称显示有些缺陷，有兴趣的小伙伴欢迎提 PR 解决~"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "关于评论模块需要注意的就是评论表的设计，把握其中字段的含义，才能透彻了解这个功能的逻辑。"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "评论 Comment 的目标类型（帖子，评论） entityType 和 entityId 以及对哪个用户进行评论/回复 targetId 是由前端传递给 DiscussPostController 的"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207150925.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [_vm._v("一个帖子的详情页需要封装的信息大概如下：")]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207151328.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "添加评论-事务管理" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#添加评论-事务管理" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 添加评论（事务管理）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207122908.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "私信列表和详情页" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#私信列表和详情页" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 私信列表和详情页")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207161130.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "发送私信-异步请求" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#发送私信-异步请求" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 发送私信（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207161500.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "点赞-异步请求" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#点赞-异步请求" } },
          [_vm._v("#")]
        ),
        _vm._v(" 点赞（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "将点赞相关信息存入 Redis 的数据结构 set 中。其中，key 命名为  "
        ),
        _c("code", [_vm._v("like:entity:entityType:entityId")]),
        _vm._v("，value 即点赞用户的 id。比如 key =  "),
        _c("code", [_vm._v("like:entity:2:246")]),
        _vm._v("  value =  "),
        _c("code", [_vm._v("11")]),
        _vm._v(" 表示用户 11 对实体类型 2 即评论进行了点赞，该评论的 id 是 246")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("某个用户的获赞数量对应的存储在 Redis 中的 key 是 "),
        _c("code", [_vm._v("like:user:userId")]),
        _vm._v("，value 就是这个用户的获赞数量")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207165837.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "我的获赞数量" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#我的获赞数量" } },
          [_vm._v("#")]
        ),
        _vm._v(" 我的获赞数量")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207170003.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "关注-异步请求" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#关注-异步请求" } },
          [_vm._v("#")]
        ),
        _vm._v(" 关注（异步请求）")
      ]),
      _vm._v(" "),
      _c("ul", [
        _c("li", [
          _vm._v(
            "若 A 关注了 B，则 A 是 B 的粉丝 Follower，B 是 A 的目标 Followee"
          )
        ]),
        _vm._v(" "),
        _c("li", [
          _vm._v(
            "关注的目标可以是用户、帖子、题目等，在实现时将这些目标抽象为实体（目前只做了关注用户）"
          )
        ])
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "将某个用户关注的实体相关信息存储在 Redis 的数据结构 zset 中：key 是 "
        ),
        _c("code", [_vm._v("followee:userId:entityType")]),
        _vm._v(" ，对应的 value 是 "),
        _c("code", [_vm._v("zset(entityId, now)")]),
        _vm._v(" ，以关注的时间进行排序。比如说 "),
        _c("code", [_vm._v("followee:111:3")]),
        _vm._v(" 对应的value "),
        _c("code", [_vm._v("(20, 2020-02-03-xxxx)")]),
        _vm._v(
          "，表明用户 111 关注了实体类型为 3 即人(用户)，该帖子的 id 是 20，关注该帖子的时间是 2020-02-03-xxxx"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "同样的，将某个实体拥有的粉丝相关信息也存储在 Redis 的数据结构 zset 中：key 是 "
        ),
        _c("code", [_vm._v("follower:entityType:entityId")]),
        _vm._v("，对应的 value 是 "),
        _c("code", [_vm._v("zset(userId, now)")]),
        _vm._v("，以关注的时间进行排序")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207174046.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "关注列表" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#关注列表" } },
          [_vm._v("#")]
        ),
        _vm._v(" 关注列表")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207175621.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "发送系统通知" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#发送系统通知" } },
          [_vm._v("#")]
        ),
        _vm._v(" 发送系统通知")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210207182917.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "显示系统通知" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#显示系统通知" } },
          [_vm._v("#")]
        ),
        _vm._v(" 显示系统通知")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208153059.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "搜索" } }, [
        _c("a", { staticClass: "header-anchor", attrs: { href: "#搜索" } }, [
          _vm._v("#")
        ]),
        _vm._v(" 搜索")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208161936.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("类似的，置顶、加精也会触发发帖事件，就不再图里面画出来了。")
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "置顶加精删除-异步请求" } }, [
        _c(
          "a",
          {
            staticClass: "header-anchor",
            attrs: { href: "#置顶加精删除-异步请求" }
          },
          [_vm._v("#")]
        ),
        _vm._v(" 置顶加精删除（异步请求）")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208171729.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "网站数据统计" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#网站数据统计" } },
          [_vm._v("#")]
        ),
        _vm._v(" 网站数据统计")
      ]),
      _vm._v(" "),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208170801.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h3", { attrs: { id: "帖子热度计算" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#帖子热度计算" } },
          [_vm._v("#")]
        ),
        _vm._v(" 帖子热度计算")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "每次发生点赞（给帖子点赞）、评论（给帖子评论）、加精的时候，就将这些帖子信息存入缓存 Redis 中，然后通过分布式的定时任务 Spring Quartz，每隔一段时间就从缓存中取出这些帖子进行计算分数。"
        )
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("帖子分数/热度计算公式：分数（热度） = 权重 + 发帖距离天数")
      ]),
      _vm._v(" "),
      _c("div", { staticClass: "language-java extra-class" }, [
        _c("pre", { pre: true, attrs: { class: "language-java" } }, [
          _c("code", [
            _c("span", { pre: true, attrs: { class: "token comment" } }, [
              _vm._v("// 计算权重")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("double")
            ]),
            _vm._v(" w "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("wonderful "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("?")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("75")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v(":")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("0")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("+")
            ]),
            _vm._v(" commentCount "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("10")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("+")
            ]),
            _vm._v(" likeCount "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("2")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token comment" } }, [
              _vm._v("// 分数 = 权重 + 发帖距离天数")
            ]),
            _vm._v("\n"),
            _c("span", { pre: true, attrs: { class: "token keyword" } }, [
              _vm._v("double")
            ]),
            _vm._v(" score "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("=")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token class-name" } }, [
              _vm._v("Math")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("log10")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token class-name" } }, [
              _vm._v("Math")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("max")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("w"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(",")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("1")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v("\n        "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("+")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _vm._v("post"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("getCreateTime")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("getTime")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("-")
            ]),
            _vm._v(" epoch"),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(".")
            ]),
            _c("span", { pre: true, attrs: { class: "token function" } }, [
              _vm._v("getTime")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("/")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v("(")
            ]),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("1000")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("3600")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token operator" } }, [
              _vm._v("*")
            ]),
            _vm._v(" "),
            _c("span", { pre: true, attrs: { class: "token number" } }, [
              _vm._v("24")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(")")
            ]),
            _c("span", { pre: true, attrs: { class: "token punctuation" } }, [
              _vm._v(";")
            ]),
            _vm._v("\n")
          ])
        ])
      ]),
      _c("p", [
        _c("img", {
          attrs: {
            src:
              "https://gitee.com/veal98/images/raw/master/img/20210208173636.png",
            alt: ""
          }
        })
      ]),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📖-配套教程" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#📖-配套教程" } },
          [_vm._v("#")]
        ),
        _vm._v(" 📖 配套教程")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v(
          "想要自己从零开始实现这个项目或者深入理解的小伙伴，可以扫描下方二维码关注公众号『"
        ),
        _c("strong", [_vm._v("飞天小牛肉")]),
        _vm._v(
          "』，第一时间获取配套教程, 不仅会详细解释本项目涉及的各大技术点，还会汇总相关的常见面试题，目前尚在更新中。"
        )
      ]),
      _vm._v(" "),
      _c("img", {
        staticStyle: { zoom: "67%" },
        attrs: {
          src:
            "https://gitee.com/veal98/images/raw/master/img/20210204145531.png"
        }
      }),
      _vm._v(" "),
      _c("h2", { attrs: { id: "📞-联系我" } }, [
        _c(
          "a",
          { staticClass: "header-anchor", attrs: { href: "#📞-联系我" } },
          [_vm._v("#")]
        ),
        _vm._v(" 📞 联系我")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("有什么问题也可以添加我的微信，记得备注来意：格式 "),
        _c("u", [_vm._v("（学校或公司 - 姓名或昵称 - 来意）")])
      ]),
      _vm._v(" "),
      _c("img", {
        attrs: {
          width: "260px",
          src:
            "https://gitee.com/veal98/images/raw/master/img/微信图片_20210105121328.jpg"
        }
      }),
      _vm._v(" "),
      _c("h2", { attrs: { id: "👏-鸣谢" } }, [
        _c("a", { staticClass: "header-anchor", attrs: { href: "#👏-鸣谢" } }, [
          _vm._v("#")
        ]),
        _vm._v(" 👏 鸣谢")
      ]),
      _vm._v(" "),
      _c("p", [
        _vm._v("本项目参考"),
        _c(
          "a",
          {
            attrs: {
              href: "https://www.nowcoder.com/",
              target: "_blank",
              rel: "noopener noreferrer"
            }
          },
          [_vm._v("牛客网"), _c("OutboundLink")],
          1
        ),
        _vm._v(" — Java 高级工程师课程，感谢老师和平台")
      ])
    ]
  )
}
var staticRenderFns = []
render._withStripped = true

export { render, staticRenderFns }"}]} \ No newline at end of file diff --git a/node_modules/@vuepress/core/node_modules/.cache/vuepress/b4c93b53937ab1eb6dd25d96729e6256.json b/node_modules/@vuepress/core/node_modules/.cache/vuepress/b4c93b53937ab1eb6dd25d96729e6256.json index 72f03fdd..b83ddde2 100644 --- a/node_modules/@vuepress/core/node_modules/.cache/vuepress/b4c93b53937ab1eb6dd25d96729e6256.json +++ b/node_modules/@vuepress/core/node_modules/.cache/vuepress/b4c93b53937ab1eb6dd25d96729e6256.json @@ -1 +1 @@ -{"remainingRequest":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js??ref--1-1!E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js??ref--1-2!E:\\GreateCommunity\\docs\\Guide.md?vue&type=template&id=1a70d707&","dependencies":[{"path":"E:\\GreateCommunity\\docs\\Guide.md","mtime":1613049784185},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\loaders\\templateLoader.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:
<ContentSlotsDistributor :slot-key="$parent.slotKey"><h1 id="echo-开源社区系统"><a class="header-anchor" href="#echo-开源社区系统">#</a> Echo — 开源社区系统</h1>
<hr>
<h2 id="📚-从本项目你能学到什么"><a class="header-anchor" href="#📚-从本项目你能学到什么">#</a> 📚 从本项目你能学到什么</h2>
<ul>
<li>学会主流的 Java Web 开发技术和框架（Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch 等）</li>
<li>了解一个真实的 Web 项目从开发到部署的整个流程（本项目配套有大量图例和详细教程，以帮助小伙伴快速上手）</li>
<li>掌握本项目中涉及的核心技术点以及常见面试题和解析</li>
</ul>
<h2 id="🏄‍-在线体验与文档地址"><a class="header-anchor" href="#🏄‍-在线体验与文档地址">#</a> 🏄‍ 在线体验与文档地址</h2>
<ul>
<li>在线体验：项目已经部署到腾讯云服务器，各位小伙伴们可直接线上体验：<a href="http://1.15.127.74/" target="_blank" rel="noopener noreferrer">http://1.15.127.74/<OutboundLink/></a></li>
<li>文档地址：文档通过 Vuepress + Gitee Pages 生成，在线访问地址：</li>
</ul>
<h2 id="💻-核心技术栈"><a class="header-anchor" href="#💻-核心技术栈">#</a> 💻 核心技术栈</h2>
<p>后端：</p>
<ul>
<li>Spring</li>
<li>Spring Boot 2.1.5 RELEASE</li>
<li>Spring MVC</li>
<li>ORM：MyBatis</li>
<li>数据库：MySQL 5.7</li>
<li>分布式缓存：Redis</li>
<li>本地缓存：Caffeine</li>
<li>消息队列：Kafka 2.13-2.7.0</li>
<li>搜索引擎：Elasticsearch 6.4.3</li>
<li>安全：Spring Security</li>
<li>邮件：Spring Mail</li>
<li>分布式定时任务：Spring Quartz</li>
<li>日志：SLF4J（日志接口） + Logback（日志实现）</li>
</ul>
<p>前端：</p>
<ul>
<li>Thymeleaf</li>
<li>Bootstrap 4.x</li>
<li>Jquery</li>
<li>Ajax</li>
</ul>
<h2 id="🔨-开发环境"><a class="header-anchor" href="#🔨-开发环境">#</a> 🔨 开发环境</h2>
<ul>
<li>操作系统：Windows 10</li>
<li>构建工具：Apache Maven</li>
<li>集成开发工具：Intellij IDEA</li>
<li>应用服务器：Apache Tomcat</li>
<li>接口测试工具：Postman</li>
<li>压力测试工具：Apache JMeter</li>
<li>版本控制工具：Git</li>
<li>Java 版本：8</li>
</ul>
<h2 id="🎀-界面展示"><a class="header-anchor" href="#🎀-界面展示">#</a> 🎀 界面展示</h2>
<p>首页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205641.png" alt=""></p>
<p>登录页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205558.png" alt=""></p>
<p>帖子详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205741.png" alt=""></p>
<p>个人主页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205820.png" alt=""></p>
<p>朋友私信页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205857.png" alt=""></p>
<p>私信详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205948.png" alt=""></p>
<p>系统通知页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210122.png" alt=""></p>
<p>通知详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210152.png" alt=""></p>
<p>账号设置页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210238.png" alt=""></p>
<p>数据统计页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210323.png" alt=""></p>
<p>搜索详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210531.png" alt=""></p>
<h2 id="🎨-功能列表"><a class="header-anchor" href="#🎨-功能列表">#</a> 🎨 功能列表</h2>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208222403.png" alt=""></p>
<ul>
<li>
<p>[x] <strong>注册</strong></p>
<ul>
<li>用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活</li>
<li>向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）</li>
</ul>
</li>
<li>
<p>[x] <strong>登录 | 登出</strong></p>
<ul>
<li>
<p>进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）</p>
</li>
<li>
<p>用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis</p>
<p>注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）</p>
</li>
<li>
<p>勾选记住我，则延长登录凭证有效时间</p>
</li>
<li>
<p>用户登录成功，将用户信息短暂存入 Redis（1 小时）</p>
</li>
<li>
<p>用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息</p>
</li>
</ul>
</li>
<li>
<p>[x] <strong>账号设置</strong></p>
<ul>
<li>修改头像
<ul>
<li>将用户选择的头像图片文件上传至七牛云服务器</li>
</ul>
</li>
<li>修改密码</li>
</ul>
</li>
<li>
<p>[x] <strong>帖子模块</strong></p>
<ul>
<li>发布帖子（过滤敏感词），将其存入 MySQL</li>
<li>分页显示所有的帖子
<ul>
<li>支持按照 “发帖时间” 显示</li>
<li>支持按照 “热度排行” 显示（Spring Quartz）</li>
</ul>
</li>
<li>查看帖子详情</li>
<li>权限管理（Spring Security + Thymeleaf Security）
<ul>
<li>未登录用户无法发帖</li>
<li>“版主” 可以看到帖子的置顶和加精按钮并执行相应操作</li>
<li>“管理员” 可以看到帖子的删除按钮并执行相应操作</li>
<li>“普通用户” 无法看到帖子的置顶、加精、删除按钮，也无法执行相应操作</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>评论模块</strong></p>
<ul>
<li>发布对帖子的评论（过滤敏感词），将其存入 MySQL</li>
<li>分页显示评论</li>
<li>发布对评论的回复（过滤敏感词）</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用评论功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>私信模块</strong></p>
<ul>
<li>发送私信（过滤敏感词）</li>
<li>私信列表
<ul>
<li>查询当前用户的会话列表</li>
<li>每个会话只显示一条最新的私信</li>
<li>支持分页显示</li>
</ul>
</li>
<li>私信详情
<ul>
<li>查询某个会话所包含的所有私信</li>
<li>访问私信详情时，将显示的私信设为已读状态</li>
<li>支持分页显示</li>
</ul>
</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用私信功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>统一处理 404 / 500 异常</strong></p>
<ul>
<li>普通请求异常</li>
<li>异步请求异常</li>
</ul>
</li>
<li>
<p>[x] <strong>统一记录日志</strong></p>
</li>
<li>
<p>[x] <strong>点赞模块</strong></p>
<ul>
<li>
<p>支持对帖子、评论/回复点赞</p>
</li>
<li>
<p>第 1 次点赞，第 2 次取消点赞</p>
</li>
<li>
<p>首页统计帖子的点赞数量</p>
</li>
<li>
<p>详情页统计帖子和评论/回复的点赞数量</p>
</li>
<li>
<p>详情页显示当前登录用户的点赞状态（赞过了则显示已赞）</p>
</li>
<li>
<p>统计我的获赞数量</p>
</li>
<li>
<p>权限管理（Spring Security）</p>
<ul>
<li>未登录用户无法使用点赞相关功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>关注模块</strong></p>
<ul>
<li>关注功能</li>
<li>取消关注功能</li>
<li>统计用户的关注数和粉丝数</li>
<li>我的关注列表（查询某个用户关注的人），支持分页</li>
<li>我的粉丝列表（查询某个用户的粉丝），支持分页</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用关注相关功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>系统通知模块</strong></p>
<ul>
<li>通知列表
<ul>
<li>显示评论、点赞、关注三种类型的通知</li>
</ul>
</li>
<li>通知详情
<ul>
<li>分页显示某一类主题所包含的通知</li>
<li>进入某种类型的系统通知详情，则将该页的所有未读的系统通知状态设置为已读</li>
</ul>
</li>
<li>未读数量
<ul>
<li>分别显示每种类型的系统通知的未读数量</li>
<li>显示所有系统通知的未读数量</li>
</ul>
</li>
<li>导航栏显示所有消息的未读数量（未读私信 + 未读系统通知）</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用系统通知功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>搜索模块</strong></p>
<ul>
<li>发布事件
<ul>
<li>发布帖子时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器</li>
<li>为帖子增加评论时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器</li>
</ul>
</li>
<li>搜索服务
<ul>
<li>从 Elasticsearch 服务器搜索帖子</li>
<li>从 Elasticsearch 服务器删除帖子（当帖子从数据库中被删除时）</li>
</ul>
</li>
<li>显示搜索结果</li>
</ul>
</li>
<li>
<p>[x] <strong>网站数据统计</strong>（管理员专属）</p>
<ul>
<li>独立访客 UV
<ul>
<li>存入 Redis 的 HyperLogLog</li>
<li>支持单日查询和区间日期查询</li>
</ul>
</li>
<li>日活跃用户 DAU
<ul>
<li>存入 Redis 的 Bitmap</li>
<li>支持单日查询和区间日期查询</li>
</ul>
</li>
<li>权限管理（Spring Security）
<ul>
<li>只有管理员可以查看网站数据统计</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] 优化网站性能</p>
<ul>
<li>使用本地缓存 Caffeine 缓存热帖列表以及所有用户帖子的总数</li>
</ul>
</li>
</ul>
<h2 id="🔐-待实现及优化"><a class="header-anchor" href="#🔐-待实现及优化">#</a> 🔐 待实现及优化</h2>
<p>以下是我个人发现的本项目存在的问题，但是暂时没有头绪无法解决，集思广益，欢迎各位小伙伴提 PR 解决：</p>
<ul>
<li>[ ] 注册模块无法正常跳转到操作提示界面（本地运行没有问题）</li>
<li>[ ] 评论功能的前端显示部分存在 Bug</li>
<li>[ ] 查询我的评论（未完善）</li>
</ul>
<p>以下是我觉得本项目还可以添加的功能，同样欢迎各位小伙伴提 issue 指出还可以增加哪些功能，或者直接提 PR 实现该功能：</p>
<ul>
<li>[ ] 忘记密码（发送邮件找回密码）</li>
<li>[ ] 查询我的点赞</li>
<li>[ ] 管理员对帖子的二次点击取消置顶功能</li>
<li>[ ] 管理员对已删除帖子的恢复功能（本项目中的删除帖子并未将其从数据库中删除，只是将其状态设置为了拉黑）</li>
</ul>
<h2 id="🌱-本地运行"><a class="header-anchor" href="#🌱-本地运行">#</a> 🌱 本地运行</h2>
<p>各位如果需要将项目部署在本地进行测试，以下环境请提前备好：</p>
<ul>
<li>Java 8</li>
<li>MySQL 5.7</li>
<li>Redis</li>
<li>Kafka 2.13-2.7.0</li>
<li>Elasticsearch 6.4.3</li>
</ul>
<p>然后<strong>修改配置文件中的信息为你自己的本地环境，直接运行是运行不了的</strong>，而且相关私密信息我全部用 xxxxxxx 代替了。</p>
<p>本地运行需要修改的配置文件信息如下：</p>
<p>1）<code>application-develop.properties</code>：</p>
<ul>
<li>MySQL</li>
<li>Spring Mail（邮箱需要开启 SMTP 服务）</li>
<li>Kafka：consumer.group-id（该字段见 Kafka 安装包中的 consumer.proerties，可自行修改, 修改完毕后需要重启 Kafka）</li>
<li>Elasticsearch：cluster-name（该字段见 Elasticsearch 安装包中的 elasticsearch.yml，可自行修改）</li>
<li>七牛云（需要新建一个七牛云的对象存储空间，用来存放上传的头像图片）</li>
</ul>
<p>2）<code>logback-spring-develop.xml</code>：</p>
<ul>
<li>LOG_PATH：日志存放的位置</li>
</ul>
<p>每次运行需要打开：</p>
<ul>
<li>MySQL</li>
<li>Redis</li>
<li>Elasticsearch</li>
<li>Kafka</li>
</ul>
<p>另外，还需要事件建好数据库表，详细见下文。</p>
<h2 id="📜-数据库设计"><a class="header-anchor" href="#📜-数据库设计">#</a> 📜 数据库设计</h2>
<p>用户 <code>user</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">DROP</span> <span class="token keyword">TABLE</span> <span class="token keyword">IF</span> <span class="token keyword">EXISTS</span> <span class="token punctuation">`</span><span class="token keyword">user</span><span class="token punctuation">`</span><span class="token punctuation">;</span>
<span class="token keyword">SET</span> character_set_client <span class="token operator">=</span> utf8mb4 <span class="token punctuation">;</span>
<span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span><span class="token keyword">user</span><span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>username<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">50</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>password<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">50</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>salt<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">50</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>email<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">100</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">type</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-普通用户; 1-超级管理员; 2-版主;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-未激活; 1-已激活;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>activation_code<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">100</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>header_url<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">200</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_username<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>username<span class="token punctuation">`</span><span class="token punctuation">(</span><span class="token number">20</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_email<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>email<span class="token punctuation">`</span><span class="token punctuation">(</span><span class="token number">20</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token operator">=</span><span class="token number">101</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p>讨论帖 <code>discuss_post</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">DROP</span> <span class="token keyword">TABLE</span> <span class="token keyword">IF</span> <span class="token keyword">EXISTS</span> <span class="token punctuation">`</span>discuss_post<span class="token punctuation">`</span><span class="token punctuation">;</span>
<span class="token keyword">SET</span> character_set_client <span class="token operator">=</span> utf8mb4 <span class="token punctuation">;</span>
<span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span>discuss_post<span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>user_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>title<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">100</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>content<span class="token punctuation">`</span> <span class="token keyword">text</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">type</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-普通; 1-置顶;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-正常; 1-精华; 2-拉黑;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>comment_count<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>score<span class="token punctuation">`</span> <span class="token keyword">double</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_user_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>user_id<span class="token punctuation">`</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p>评论（回复）<code>comment</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span><span class="token keyword">comment</span><span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>user_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>entity_type<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'评论目标的类别：1 帖子；2 评论 '</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>entity_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'评论目标的 id'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>target_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'指明对谁进行评论'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>content<span class="token punctuation">`</span> <span class="token keyword">text</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'状态：0 正常；1 禁用'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_user_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>user_id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_entity_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>entity_id<span class="token punctuation">`</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token operator">=</span><span class="token number">247</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p>私信 <code>message</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">DROP</span> <span class="token keyword">TABLE</span> <span class="token keyword">IF</span> <span class="token keyword">EXISTS</span> <span class="token punctuation">`</span>message<span class="token punctuation">`</span><span class="token punctuation">;</span>
<span class="token keyword">SET</span> character_set_client <span class="token operator">=</span> utf8mb4 <span class="token punctuation">;</span>
<span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span>message<span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>from_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>to_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>conversation_id<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">45</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>content<span class="token punctuation">`</span> <span class="token keyword">text</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-未读;1-已读;2-删除;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_from_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>from_id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_to_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>to_id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_conversation_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>conversation_id<span class="token punctuation">`</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><h2 id="🌌-理想的部署架构"><a class="header-anchor" href="#🌌-理想的部署架构">#</a> 🌌 理想的部署架构</h2>
<p>我每个都只部署了一台，以下是理想的部署架构：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211204207.png" alt=""></p>
<h2 id="🎯-功能逻辑图"><a class="header-anchor" href="#🎯-功能逻辑图">#</a> 🎯 功能逻辑图</h2>
<p>画了一些不是那么严谨的图帮助各位小伙伴理清思绪。</p>
<blockquote>
<p>单向绿色箭头：</p>
<ul>
<li>前端模板 -&gt; Controller：表示这个前端模板中有一个超链接是由这个 Controller 处理的</li>
<li>Controller -&gt; 前端模板：表示这个 Controller 会像该前端模板传递数据或者跳转</li>
</ul>
<p>双向绿色箭头：表示 Controller 和前端模板之间进行参数的相互传递或使用</p>
<p>单向蓝色箭头： A -&gt; B，表示 A 方法调用了 B 方法</p>
<p>单向红色箭头：数据库或缓存操作</p>
</blockquote>
<h3 id="注册"><a class="header-anchor" href="#注册">#</a> 注册</h3>
<ul>
<li>用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活</li>
<li>向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）</li>
</ul>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210204222249.png" alt=""></p>
<h3 id="登录-登出"><a class="header-anchor" href="#登录-登出">#</a> 登录 | 登出</h3>
<ul>
<li>
<p>进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）</p>
</li>
<li>
<p>用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis</p>
<p>注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）</p>
</li>
<li>
<p>勾选记住我，则延长登录凭证有效时间</p>
</li>
<li>
<p>用户登录成功，将用户信息短暂存入 Redis（1 小时）</p>
</li>
<li>
<p>用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息</p>
</li>
</ul>
<p>下图是登录模块的功能逻辑图，并没有使用 Spring Security 提供的认证逻辑（我觉得这个模块是最复杂的，这张图其实很多细节还没有画全）</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210204233233.png" alt=""></p>
<h3 id="分页显示所有的帖子"><a class="header-anchor" href="#分页显示所有的帖子">#</a> 分页显示所有的帖子</h3>
<ul>
<li>支持按照 “发帖时间” 显示</li>
<li>支持按照 “热度排行” 显示（Spring Quartz）</li>
<li>将热帖列表和所有帖子的总数存入本地缓存 Caffeine（利用分布式定时任务 Spring Quartz 每隔一段时间就刷新计算帖子的热度/分数 — 见下文，而 Caffeine 里的数据更新不用我们操心，它天生就会自动的更新它拥有的数据，给它一个初始化方法就完事儿）</li>
</ul>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210204222822.png" alt=""></p>
<h3 id="账号设置"><a class="header-anchor" href="#账号设置">#</a> 账号设置</h3>
<ul>
<li>修改头像（异步请求）
<ul>
<li>将用户选择的头像图片文件上传至七牛云服务器</li>
</ul>
</li>
<li>修改密码</li>
</ul>
<p>此处只画出修改头像：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210206121201.png" alt=""></p>
<h3 id="发布帖子-异步请求"><a class="header-anchor" href="#发布帖子-异步请求">#</a> 发布帖子（异步请求）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210206122521.png" alt=""></p>
<h3 id="显示评论及相关信息"><a class="header-anchor" href="#显示评论及相关信息">#</a> 显示评论及相关信息</h3>
<blockquote>
<p>评论部分前端的名称显示有些缺陷，有兴趣的小伙伴欢迎提 PR 解决~</p>
</blockquote>
<p>关于评论模块需要注意的就是评论表的设计，把握其中字段的含义，才能透彻了解这个功能的逻辑。</p>
<p>评论 Comment 的目标类型（帖子，评论） entityType 和 entityId 以及对哪个用户进行评论/回复 targetId 是由前端传递给 DiscussPostController 的</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207150925.png" alt=""></p>
<p>一个帖子的详情页需要封装的信息大概如下：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207151328.png" alt=""></p>
<h3 id="添加评论-事务管理"><a class="header-anchor" href="#添加评论-事务管理">#</a> 添加评论（事务管理）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207122908.png" alt=""></p>
<h3 id="私信列表和详情页"><a class="header-anchor" href="#私信列表和详情页">#</a> 私信列表和详情页</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207161130.png" alt=""></p>
<h3 id="发送私信-异步请求"><a class="header-anchor" href="#发送私信-异步请求">#</a> 发送私信（异步请求）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207161500.png" alt=""></p>
<h3 id="点赞-异步请求"><a class="header-anchor" href="#点赞-异步请求">#</a> 点赞（异步请求）</h3>
<p>将点赞相关信息存入 Redis 的数据结构 set 中。其中，key 命名为  <code>like:entity:entityType:entityId</code>，value 即点赞用户的 id。比如 key =  <code>like:entity:2:246</code>  value =  <code>11</code> 表示用户 11 对实体类型 2 即评论进行了点赞，该评论的 id 是 246</p>
<p>某个用户的获赞数量对应的存储在 Redis 中的 key 是 <code>like:user:userId</code>，value 就是这个用户的获赞数量</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207165837.png" alt=""></p>
<h3 id="我的获赞数量"><a class="header-anchor" href="#我的获赞数量">#</a> 我的获赞数量</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207170003.png" alt=""></p>
<h3 id="关注-异步请求"><a class="header-anchor" href="#关注-异步请求">#</a> 关注（异步请求）</h3>
<ul>
<li>若 A 关注了 B，则 A 是 B 的粉丝 Follower，B 是 A 的目标 Followee</li>
<li>关注的目标可以是用户、帖子、题目等，在实现时将这些目标抽象为实体（目前只做了关注用户）</li>
</ul>
<p>将某个用户关注的实体相关信息存储在 Redis 的数据结构 zset 中：key 是 <code>followee:userId:entityType</code> ，对应的 value 是 <code>zset(entityId, now)</code> ，以关注的时间进行排序。比如说 <code>followee:111:3</code> 对应的value <code>(20, 2020-02-03-xxxx)</code>，表明用户 111 关注了实体类型为 3 即人(用户)，该帖子的 id 是 20，关注该帖子的时间是 2020-02-03-xxxx</p>
<p>同样的，将某个实体拥有的粉丝相关信息也存储在 Redis 的数据结构 zset 中：key 是 <code>follower:entityType:entityId</code>，对应的 value 是 <code>zset(userId, now)</code>，以关注的时间进行排序</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207174046.png" alt=""></p>
<h3 id="关注列表"><a class="header-anchor" href="#关注列表">#</a> 关注列表</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207175621.png" alt=""></p>
<h3 id="发送系统通知"><a class="header-anchor" href="#发送系统通知">#</a> 发送系统通知</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207182917.png" alt=""></p>
<h3 id="显示系统通知"><a class="header-anchor" href="#显示系统通知">#</a> 显示系统通知</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208153059.png" alt=""></p>
<h3 id="搜索"><a class="header-anchor" href="#搜索">#</a> 搜索</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208161936.png" alt=""></p>
<p>类似的，置顶、加精也会触发发帖事件，就不再图里面画出来了。</p>
<h3 id="置顶加精删除-异步请求"><a class="header-anchor" href="#置顶加精删除-异步请求">#</a> 置顶加精删除（异步请求）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208171729.png" alt=""></p>
<h3 id="网站数据统计"><a class="header-anchor" href="#网站数据统计">#</a> 网站数据统计</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208170801.png" alt=""></p>
<h3 id="帖子热度计算"><a class="header-anchor" href="#帖子热度计算">#</a> 帖子热度计算</h3>
<p>每次发生点赞（给帖子点赞）、评论（给帖子评论）、加精的时候，就将这些帖子信息存入缓存 Redis 中，然后通过分布式的定时任务 Spring Quartz，每隔一段时间就从缓存中取出这些帖子进行计算分数。</p>
<p>帖子分数/热度计算公式：分数（热度） = 权重 + 发帖距离天数</p>
<!--beforebegin--><div class="language-java extra-class"><!--afterbegin--><pre v-pre class="language-java"><code><span class="token comment">// 计算权重</span>
<span class="token keyword">double</span> w <span class="token operator">=</span> <span class="token punctuation">(</span>wonderful <span class="token operator">?</span> <span class="token number">75</span> <span class="token operator">:</span> <span class="token number">0</span><span class="token punctuation">)</span> <span class="token operator">+</span> commentCount <span class="token operator">*</span> <span class="token number">10</span> <span class="token operator">+</span> likeCount <span class="token operator">*</span> <span class="token number">2</span><span class="token punctuation">;</span>
<span class="token comment">// 分数 = 权重 + 发帖距离天数</span>
<span class="token keyword">double</span> score <span class="token operator">=</span> <span class="token class-name">Math</span><span class="token punctuation">.</span><span class="token function">log10</span><span class="token punctuation">(</span><span class="token class-name">Math</span><span class="token punctuation">.</span><span class="token function">max</span><span class="token punctuation">(</span>w<span class="token punctuation">,</span> <span class="token number">1</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
        <span class="token operator">+</span> <span class="token punctuation">(</span>post<span class="token punctuation">.</span><span class="token function">getCreateTime</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">getTime</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-</span> epoch<span class="token punctuation">.</span><span class="token function">getTime</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span> <span class="token operator">/</span> <span class="token punctuation">(</span><span class="token number">1000</span> <span class="token operator">*</span> <span class="token number">3600</span> <span class="token operator">*</span> <span class="token number">24</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p><img src="https://gitee.com/veal98/images/raw/master/img/20210208173636.png" alt=""></p>
<h2 id="📖-配套教程"><a class="header-anchor" href="#📖-配套教程">#</a> 📖 配套教程</h2>
<p>想要自己从零开始实现这个项目或者深入理解的小伙伴，可以扫描下方二维码关注公众号『<strong>飞天小牛肉</strong>』，第一时间获取配套教程, 不仅会详细解释本项目涉及的各大技术点，还会汇总相关的常见面试题，目前尚在更新中。</p>
<img src="https://gitee.com/veal98/images/raw/master/img/20210204145531.png" style="zoom:67%;" />
<h2 id="📞-联系我"><a class="header-anchor" href="#📞-联系我">#</a> 📞 联系我</h2>
<p>有什么问题也可以添加我的微信，记得备注来意：格式 <u>（学校或公司 - 姓名或昵称 - 来意）</u></p>
<img width="260px" src="https://gitee.com/veal98/images/raw/master/img/微信图片_20210105121328.jpg" >
<h2 id="👏-鸣谢"><a class="header-anchor" href="#👏-鸣谢">#</a> 👏 鸣谢</h2>
<p>本项目参考<a href="https://www.nowcoder.com/" target="_blank" rel="noopener noreferrer">牛客网<OutboundLink/></a> — Java 高级工程师课程，感谢老师和平台</p>
</ContentSlotsDistributor>
"},null]} \ No newline at end of file +{"remainingRequest":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js??ref--1-1!E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js??ref--1-2!E:\\GreateCommunity\\docs\\Guide.md?vue&type=template&id=1a70d707&","dependencies":[{"path":"E:\\GreateCommunity\\docs\\Guide.md","mtime":1613050811891},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\loaders\\templateLoader.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:
<ContentSlotsDistributor :slot-key="$parent.slotKey"><h1 id="echo-开源社区系统"><a class="header-anchor" href="#echo-开源社区系统">#</a> Echo — 开源社区系统</h1>
<hr>
<h2 id="📚-从本项目你能学到什么"><a class="header-anchor" href="#📚-从本项目你能学到什么">#</a> 📚 从本项目你能学到什么</h2>
<ul>
<li>学会主流的 Java Web 开发技术和框架（Spring、SpringBoot、Spring MVC、MyBatis、MySQL、Redis、Kafka、Elasticsearch、Spring Security 等）</li>
<li>了解一个真实的 Web 项目从开发到部署的整个流程（本项目配套有大量图例和详细教程，以帮助小伙伴快速上手）</li>
<li>掌握本项目中涉及的核心技术点以及常见面试题和解析</li>
</ul>
<h2 id="🏄‍-在线体验与文档地址"><a class="header-anchor" href="#🏄‍-在线体验与文档地址">#</a> 🏄‍ 在线体验与文档地址</h2>
<ul>
<li>在线体验：项目已经部署到腾讯云服务器，各位小伙伴们可直接线上体验：<a href="http://1.15.127.74/" target="_blank" rel="noopener noreferrer">http://1.15.127.74/<OutboundLink/></a></li>
<li>文档地址：文档通过 Vuepress + Gitee Pages 生成，在线访问地址：</li>
</ul>
<h2 id="💻-核心技术栈"><a class="header-anchor" href="#💻-核心技术栈">#</a> 💻 核心技术栈</h2>
<p>后端：</p>
<ul>
<li>Spring</li>
<li>Spring Boot 2.1.5 RELEASE</li>
<li>Spring MVC</li>
<li>ORM：MyBatis</li>
<li>数据库：MySQL 5.7</li>
<li>分布式缓存：Redis</li>
<li>本地缓存：Caffeine</li>
<li>消息队列：Kafka 2.13-2.7.0</li>
<li>搜索引擎：Elasticsearch 6.4.3</li>
<li>安全：Spring Security</li>
<li>邮件：Spring Mail</li>
<li>分布式定时任务：Spring Quartz</li>
<li>日志：SLF4J（日志接口） + Logback（日志实现）</li>
</ul>
<p>前端：</p>
<ul>
<li>Thymeleaf</li>
<li>Bootstrap 4.x</li>
<li>Jquery</li>
<li>Ajax</li>
</ul>
<h2 id="🔨-开发环境"><a class="header-anchor" href="#🔨-开发环境">#</a> 🔨 开发环境</h2>
<ul>
<li>操作系统：Windows 10</li>
<li>构建工具：Apache Maven</li>
<li>集成开发工具：Intellij IDEA</li>
<li>应用服务器：Apache Tomcat</li>
<li>接口测试工具：Postman</li>
<li>压力测试工具：Apache JMeter</li>
<li>版本控制工具：Git</li>
<li>Java 版本：8</li>
</ul>
<h2 id="🎀-界面展示"><a class="header-anchor" href="#🎀-界面展示">#</a> 🎀 界面展示</h2>
<p>首页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205641.png" alt=""></p>
<p>登录页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205558.png" alt=""></p>
<p>帖子详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205741.png" alt=""></p>
<p>个人主页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205820.png" alt=""></p>
<p>朋友私信页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205857.png" alt=""></p>
<p>私信详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211205948.png" alt=""></p>
<p>系统通知页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210122.png" alt=""></p>
<p>通知详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210152.png" alt=""></p>
<p>账号设置页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210238.png" alt=""></p>
<p>数据统计页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210323.png" alt=""></p>
<p>搜索详情页：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211210531.png" alt=""></p>
<h2 id="🎨-功能列表"><a class="header-anchor" href="#🎨-功能列表">#</a> 🎨 功能列表</h2>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208222403.png" alt=""></p>
<ul>
<li>
<p>[x] <strong>注册</strong></p>
<ul>
<li>用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活</li>
<li>向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）</li>
</ul>
</li>
<li>
<p>[x] <strong>登录 | 登出</strong></p>
<ul>
<li>
<p>进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）</p>
</li>
<li>
<p>用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis</p>
<p>注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）</p>
</li>
<li>
<p>勾选记住我，则延长登录凭证有效时间</p>
</li>
<li>
<p>用户登录成功，将用户信息短暂存入 Redis（1 小时）</p>
</li>
<li>
<p>用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息</p>
</li>
</ul>
</li>
<li>
<p>[x] <strong>账号设置</strong></p>
<ul>
<li>修改头像
<ul>
<li>将用户选择的头像图片文件上传至七牛云服务器</li>
</ul>
</li>
<li>修改密码</li>
</ul>
</li>
<li>
<p>[x] <strong>帖子模块</strong></p>
<ul>
<li>发布帖子（过滤敏感词），将其存入 MySQL</li>
<li>分页显示所有的帖子
<ul>
<li>支持按照 “发帖时间” 显示</li>
<li>支持按照 “热度排行” 显示（Spring Quartz）</li>
</ul>
</li>
<li>查看帖子详情</li>
<li>权限管理（Spring Security + Thymeleaf Security）
<ul>
<li>未登录用户无法发帖</li>
<li>“版主” 可以看到帖子的置顶和加精按钮并执行相应操作</li>
<li>“管理员” 可以看到帖子的删除按钮并执行相应操作</li>
<li>“普通用户” 无法看到帖子的置顶、加精、删除按钮，也无法执行相应操作</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>评论模块</strong></p>
<ul>
<li>发布对帖子的评论（过滤敏感词），将其存入 MySQL</li>
<li>分页显示评论</li>
<li>发布对评论的回复（过滤敏感词）</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用评论功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>私信模块</strong></p>
<ul>
<li>发送私信（过滤敏感词）</li>
<li>私信列表
<ul>
<li>查询当前用户的会话列表</li>
<li>每个会话只显示一条最新的私信</li>
<li>支持分页显示</li>
</ul>
</li>
<li>私信详情
<ul>
<li>查询某个会话所包含的所有私信</li>
<li>访问私信详情时，将显示的私信设为已读状态</li>
<li>支持分页显示</li>
</ul>
</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用私信功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>统一处理 404 / 500 异常</strong></p>
<ul>
<li>普通请求异常</li>
<li>异步请求异常</li>
</ul>
</li>
<li>
<p>[x] <strong>统一记录日志</strong></p>
</li>
<li>
<p>[x] <strong>点赞模块</strong></p>
<ul>
<li>
<p>支持对帖子、评论/回复点赞</p>
</li>
<li>
<p>第 1 次点赞，第 2 次取消点赞</p>
</li>
<li>
<p>首页统计帖子的点赞数量</p>
</li>
<li>
<p>详情页统计帖子和评论/回复的点赞数量</p>
</li>
<li>
<p>详情页显示当前登录用户的点赞状态（赞过了则显示已赞）</p>
</li>
<li>
<p>统计我的获赞数量</p>
</li>
<li>
<p>权限管理（Spring Security）</p>
<ul>
<li>未登录用户无法使用点赞相关功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>关注模块</strong></p>
<ul>
<li>关注功能</li>
<li>取消关注功能</li>
<li>统计用户的关注数和粉丝数</li>
<li>我的关注列表（查询某个用户关注的人），支持分页</li>
<li>我的粉丝列表（查询某个用户的粉丝），支持分页</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用关注相关功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>系统通知模块</strong></p>
<ul>
<li>通知列表
<ul>
<li>显示评论、点赞、关注三种类型的通知</li>
</ul>
</li>
<li>通知详情
<ul>
<li>分页显示某一类主题所包含的通知</li>
<li>进入某种类型的系统通知详情，则将该页的所有未读的系统通知状态设置为已读</li>
</ul>
</li>
<li>未读数量
<ul>
<li>分别显示每种类型的系统通知的未读数量</li>
<li>显示所有系统通知的未读数量</li>
</ul>
</li>
<li>导航栏显示所有消息的未读数量（未读私信 + 未读系统通知）</li>
<li>权限管理（Spring Security）
<ul>
<li>未登录用户无法使用系统通知功能</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] <strong>搜索模块</strong></p>
<ul>
<li>发布事件
<ul>
<li>发布帖子时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器</li>
<li>为帖子增加评论时，通过消息队列将帖子异步地提交到 Elasticsearch 服务器</li>
</ul>
</li>
<li>搜索服务
<ul>
<li>从 Elasticsearch 服务器搜索帖子</li>
<li>从 Elasticsearch 服务器删除帖子（当帖子从数据库中被删除时）</li>
</ul>
</li>
<li>显示搜索结果</li>
</ul>
</li>
<li>
<p>[x] <strong>网站数据统计</strong>（管理员专属）</p>
<ul>
<li>独立访客 UV
<ul>
<li>存入 Redis 的 HyperLogLog</li>
<li>支持单日查询和区间日期查询</li>
</ul>
</li>
<li>日活跃用户 DAU
<ul>
<li>存入 Redis 的 Bitmap</li>
<li>支持单日查询和区间日期查询</li>
</ul>
</li>
<li>权限管理（Spring Security）
<ul>
<li>只有管理员可以查看网站数据统计</li>
</ul>
</li>
</ul>
</li>
<li>
<p>[x] 优化网站性能</p>
<ul>
<li>使用本地缓存 Caffeine 缓存热帖列表以及所有用户帖子的总数</li>
</ul>
</li>
</ul>
<h2 id="🔐-待实现及优化"><a class="header-anchor" href="#🔐-待实现及优化">#</a> 🔐 待实现及优化</h2>
<p>以下是我个人发现的本项目存在的问题，但是暂时没有头绪无法解决，集思广益，欢迎各位小伙伴提 PR 解决：</p>
<ul>
<li>[ ] 注册模块无法正常跳转到操作提示界面（本地运行没有问题）</li>
<li>[ ] 评论功能的前端显示部分存在 Bug</li>
<li>[ ] 查询我的评论（未完善）</li>
</ul>
<p>以下是我觉得本项目还可以添加的功能，同样欢迎各位小伙伴提 issue 指出还可以增加哪些功能，或者直接提 PR 实现该功能：</p>
<ul>
<li>[ ] 忘记密码（发送邮件找回密码）</li>
<li>[ ] 查询我的点赞</li>
<li>[ ] 管理员对帖子的二次点击取消置顶功能</li>
<li>[ ] 管理员对已删除帖子的恢复功能（本项目中的删除帖子并未将其从数据库中删除，只是将其状态设置为了拉黑）</li>
</ul>
<h2 id="🌱-本地运行"><a class="header-anchor" href="#🌱-本地运行">#</a> 🌱 本地运行</h2>
<p>各位如果需要将项目部署在本地进行测试，以下环境请提前备好：</p>
<ul>
<li>Java 8</li>
<li>MySQL 5.7</li>
<li>Redis</li>
<li>Kafka 2.13-2.7.0</li>
<li>Elasticsearch 6.4.3</li>
</ul>
<p>然后<strong>修改配置文件中的信息为你自己的本地环境，直接运行是运行不了的</strong>，而且相关私密信息我全部用 xxxxxxx 代替了。</p>
<p>本地运行需要修改的配置文件信息如下：</p>
<p>1）<code>application-develop.properties</code>：</p>
<ul>
<li>MySQL</li>
<li>Spring Mail（邮箱需要开启 SMTP 服务）</li>
<li>Kafka：consumer.group-id（该字段见 Kafka 安装包中的 consumer.proerties，可自行修改, 修改完毕后需要重启 Kafka）</li>
<li>Elasticsearch：cluster-name（该字段见 Elasticsearch 安装包中的 elasticsearch.yml，可自行修改）</li>
<li>七牛云（需要新建一个七牛云的对象存储空间，用来存放上传的头像图片）</li>
</ul>
<p>2）<code>logback-spring-develop.xml</code>：</p>
<ul>
<li>LOG_PATH：日志存放的位置</li>
</ul>
<p>每次运行需要打开：</p>
<ul>
<li>MySQL</li>
<li>Redis</li>
<li>Elasticsearch</li>
<li>Kafka</li>
</ul>
<p>另外，还需要事件建好数据库表，详细见下文。</p>
<h2 id="📜-数据库设计"><a class="header-anchor" href="#📜-数据库设计">#</a> 📜 数据库设计</h2>
<p>用户 <code>user</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">DROP</span> <span class="token keyword">TABLE</span> <span class="token keyword">IF</span> <span class="token keyword">EXISTS</span> <span class="token punctuation">`</span><span class="token keyword">user</span><span class="token punctuation">`</span><span class="token punctuation">;</span>
<span class="token keyword">SET</span> character_set_client <span class="token operator">=</span> utf8mb4 <span class="token punctuation">;</span>
<span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span><span class="token keyword">user</span><span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>username<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">50</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>password<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">50</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>salt<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">50</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>email<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">100</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">type</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-普通用户; 1-超级管理员; 2-版主;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-未激活; 1-已激活;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>activation_code<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">100</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>header_url<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">200</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_username<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>username<span class="token punctuation">`</span><span class="token punctuation">(</span><span class="token number">20</span><span class="token punctuation">)</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_email<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>email<span class="token punctuation">`</span><span class="token punctuation">(</span><span class="token number">20</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token operator">=</span><span class="token number">101</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p>讨论帖 <code>discuss_post</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">DROP</span> <span class="token keyword">TABLE</span> <span class="token keyword">IF</span> <span class="token keyword">EXISTS</span> <span class="token punctuation">`</span>discuss_post<span class="token punctuation">`</span><span class="token punctuation">;</span>
<span class="token keyword">SET</span> character_set_client <span class="token operator">=</span> utf8mb4 <span class="token punctuation">;</span>
<span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span>discuss_post<span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>user_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>title<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">100</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>content<span class="token punctuation">`</span> <span class="token keyword">text</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">type</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-普通; 1-置顶;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-正常; 1-精华; 2-拉黑;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>comment_count<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>score<span class="token punctuation">`</span> <span class="token keyword">double</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_user_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>user_id<span class="token punctuation">`</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p>评论（回复）<code>comment</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span><span class="token keyword">comment</span><span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>user_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>entity_type<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'评论目标的类别：1 帖子；2 评论 '</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>entity_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'评论目标的 id'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>target_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'指明对谁进行评论'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>content<span class="token punctuation">`</span> <span class="token keyword">text</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'状态：0 正常；1 禁用'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_user_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>user_id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_entity_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>entity_id<span class="token punctuation">`</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token operator">=</span><span class="token number">247</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p>私信 <code>message</code>：</p>
<!--beforebegin--><div class="language-sql extra-class"><!--afterbegin--><pre v-pre class="language-sql"><code><span class="token keyword">DROP</span> <span class="token keyword">TABLE</span> <span class="token keyword">IF</span> <span class="token keyword">EXISTS</span> <span class="token punctuation">`</span>message<span class="token punctuation">`</span><span class="token punctuation">;</span>
<span class="token keyword">SET</span> character_set_client <span class="token operator">=</span> utf8mb4 <span class="token punctuation">;</span>
<span class="token keyword">CREATE</span> <span class="token keyword">TABLE</span> <span class="token punctuation">`</span>message<span class="token punctuation">`</span> <span class="token punctuation">(</span>
  <span class="token punctuation">`</span>id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span> <span class="token keyword">AUTO_INCREMENT</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>from_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>to_id<span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>conversation_id<span class="token punctuation">`</span> <span class="token keyword">varchar</span><span class="token punctuation">(</span><span class="token number">45</span><span class="token punctuation">)</span> <span class="token operator">NOT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>content<span class="token punctuation">`</span> <span class="token keyword">text</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span><span class="token keyword">status</span><span class="token punctuation">`</span> <span class="token keyword">int</span><span class="token punctuation">(</span><span class="token number">11</span><span class="token punctuation">)</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span> <span class="token keyword">COMMENT</span> <span class="token string">'0-未读;1-已读;2-删除;'</span><span class="token punctuation">,</span>
  <span class="token punctuation">`</span>create_time<span class="token punctuation">`</span> <span class="token keyword">timestamp</span> <span class="token boolean">NULL</span> <span class="token keyword">DEFAULT</span> <span class="token boolean">NULL</span><span class="token punctuation">,</span>
  <span class="token keyword">PRIMARY</span> <span class="token keyword">KEY</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_from_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>from_id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_to_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>to_id<span class="token punctuation">`</span><span class="token punctuation">)</span><span class="token punctuation">,</span>
  <span class="token keyword">KEY</span> <span class="token punctuation">`</span>index_conversation_id<span class="token punctuation">`</span> <span class="token punctuation">(</span><span class="token punctuation">`</span>conversation_id<span class="token punctuation">`</span><span class="token punctuation">)</span>
<span class="token punctuation">)</span> <span class="token keyword">ENGINE</span><span class="token operator">=</span><span class="token keyword">InnoDB</span> <span class="token keyword">DEFAULT</span> <span class="token keyword">CHARSET</span><span class="token operator">=</span>utf8<span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><h2 id="🌌-理想的部署架构"><a class="header-anchor" href="#🌌-理想的部署架构">#</a> 🌌 理想的部署架构</h2>
<p>我每个都只部署了一台，以下是理想的部署架构：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210211204207.png" alt=""></p>
<h2 id="🎯-功能逻辑图"><a class="header-anchor" href="#🎯-功能逻辑图">#</a> 🎯 功能逻辑图</h2>
<p>画了一些不是那么严谨的图帮助各位小伙伴理清思绪。</p>
<blockquote>
<p>单向绿色箭头：</p>
<ul>
<li>前端模板 -&gt; Controller：表示这个前端模板中有一个超链接是由这个 Controller 处理的</li>
<li>Controller -&gt; 前端模板：表示这个 Controller 会像该前端模板传递数据或者跳转</li>
</ul>
<p>双向绿色箭头：表示 Controller 和前端模板之间进行参数的相互传递或使用</p>
<p>单向蓝色箭头： A -&gt; B，表示 A 方法调用了 B 方法</p>
<p>单向红色箭头：数据库或缓存操作</p>
</blockquote>
<h3 id="注册"><a class="header-anchor" href="#注册">#</a> 注册</h3>
<ul>
<li>用户注册成功，将用户信息存入 MySQL，但此时该用户状态为未激活</li>
<li>向用户发送激活邮件，用户点击链接则激活账号（Spring Mail）</li>
</ul>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210204222249.png" alt=""></p>
<h3 id="登录-登出"><a class="header-anchor" href="#登录-登出">#</a> 登录 | 登出</h3>
<ul>
<li>
<p>进入登录界面，动态生成验证码，并将验证码短暂存入 Redis（60 秒）</p>
</li>
<li>
<p>用户登录成功（验证用户名、密码、验证码），生成登录凭证且设置状态为有效，并将登录凭证存入 Redis</p>
<p>注意：登录凭证存在有效期，在所有的请求执行之前，都会检查凭证是否有效和是否过期，只要该用户的凭证有效并在有效期时间内，本次请求就会一直持有该用户信息（使用 ThreadLocal 持有用户信息）</p>
</li>
<li>
<p>勾选记住我，则延长登录凭证有效时间</p>
</li>
<li>
<p>用户登录成功，将用户信息短暂存入 Redis（1 小时）</p>
</li>
<li>
<p>用户登出，将凭证状态设为无效，并更新 Redis 中该用户的登录凭证信息</p>
</li>
</ul>
<p>下图是登录模块的功能逻辑图，并没有使用 Spring Security 提供的认证逻辑（我觉得这个模块是最复杂的，这张图其实很多细节还没有画全）</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210204233233.png" alt=""></p>
<h3 id="分页显示所有的帖子"><a class="header-anchor" href="#分页显示所有的帖子">#</a> 分页显示所有的帖子</h3>
<ul>
<li>支持按照 “发帖时间” 显示</li>
<li>支持按照 “热度排行” 显示（Spring Quartz）</li>
<li>将热帖列表和所有帖子的总数存入本地缓存 Caffeine（利用分布式定时任务 Spring Quartz 每隔一段时间就刷新计算帖子的热度/分数 — 见下文，而 Caffeine 里的数据更新不用我们操心，它天生就会自动的更新它拥有的数据，给它一个初始化方法就完事儿）</li>
</ul>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210204222822.png" alt=""></p>
<h3 id="账号设置"><a class="header-anchor" href="#账号设置">#</a> 账号设置</h3>
<ul>
<li>修改头像（异步请求）
<ul>
<li>将用户选择的头像图片文件上传至七牛云服务器</li>
</ul>
</li>
<li>修改密码</li>
</ul>
<p>此处只画出修改头像：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210206121201.png" alt=""></p>
<h3 id="发布帖子-异步请求"><a class="header-anchor" href="#发布帖子-异步请求">#</a> 发布帖子（异步请求）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210206122521.png" alt=""></p>
<h3 id="显示评论及相关信息"><a class="header-anchor" href="#显示评论及相关信息">#</a> 显示评论及相关信息</h3>
<blockquote>
<p>评论部分前端的名称显示有些缺陷，有兴趣的小伙伴欢迎提 PR 解决~</p>
</blockquote>
<p>关于评论模块需要注意的就是评论表的设计，把握其中字段的含义，才能透彻了解这个功能的逻辑。</p>
<p>评论 Comment 的目标类型（帖子，评论） entityType 和 entityId 以及对哪个用户进行评论/回复 targetId 是由前端传递给 DiscussPostController 的</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207150925.png" alt=""></p>
<p>一个帖子的详情页需要封装的信息大概如下：</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207151328.png" alt=""></p>
<h3 id="添加评论-事务管理"><a class="header-anchor" href="#添加评论-事务管理">#</a> 添加评论（事务管理）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207122908.png" alt=""></p>
<h3 id="私信列表和详情页"><a class="header-anchor" href="#私信列表和详情页">#</a> 私信列表和详情页</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207161130.png" alt=""></p>
<h3 id="发送私信-异步请求"><a class="header-anchor" href="#发送私信-异步请求">#</a> 发送私信（异步请求）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207161500.png" alt=""></p>
<h3 id="点赞-异步请求"><a class="header-anchor" href="#点赞-异步请求">#</a> 点赞（异步请求）</h3>
<p>将点赞相关信息存入 Redis 的数据结构 set 中。其中，key 命名为  <code>like:entity:entityType:entityId</code>，value 即点赞用户的 id。比如 key =  <code>like:entity:2:246</code>  value =  <code>11</code> 表示用户 11 对实体类型 2 即评论进行了点赞，该评论的 id 是 246</p>
<p>某个用户的获赞数量对应的存储在 Redis 中的 key 是 <code>like:user:userId</code>，value 就是这个用户的获赞数量</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207165837.png" alt=""></p>
<h3 id="我的获赞数量"><a class="header-anchor" href="#我的获赞数量">#</a> 我的获赞数量</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207170003.png" alt=""></p>
<h3 id="关注-异步请求"><a class="header-anchor" href="#关注-异步请求">#</a> 关注（异步请求）</h3>
<ul>
<li>若 A 关注了 B，则 A 是 B 的粉丝 Follower，B 是 A 的目标 Followee</li>
<li>关注的目标可以是用户、帖子、题目等，在实现时将这些目标抽象为实体（目前只做了关注用户）</li>
</ul>
<p>将某个用户关注的实体相关信息存储在 Redis 的数据结构 zset 中：key 是 <code>followee:userId:entityType</code> ，对应的 value 是 <code>zset(entityId, now)</code> ，以关注的时间进行排序。比如说 <code>followee:111:3</code> 对应的value <code>(20, 2020-02-03-xxxx)</code>，表明用户 111 关注了实体类型为 3 即人(用户)，该帖子的 id 是 20，关注该帖子的时间是 2020-02-03-xxxx</p>
<p>同样的，将某个实体拥有的粉丝相关信息也存储在 Redis 的数据结构 zset 中：key 是 <code>follower:entityType:entityId</code>，对应的 value 是 <code>zset(userId, now)</code>，以关注的时间进行排序</p>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207174046.png" alt=""></p>
<h3 id="关注列表"><a class="header-anchor" href="#关注列表">#</a> 关注列表</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207175621.png" alt=""></p>
<h3 id="发送系统通知"><a class="header-anchor" href="#发送系统通知">#</a> 发送系统通知</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210207182917.png" alt=""></p>
<h3 id="显示系统通知"><a class="header-anchor" href="#显示系统通知">#</a> 显示系统通知</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208153059.png" alt=""></p>
<h3 id="搜索"><a class="header-anchor" href="#搜索">#</a> 搜索</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208161936.png" alt=""></p>
<p>类似的，置顶、加精也会触发发帖事件，就不再图里面画出来了。</p>
<h3 id="置顶加精删除-异步请求"><a class="header-anchor" href="#置顶加精删除-异步请求">#</a> 置顶加精删除（异步请求）</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208171729.png" alt=""></p>
<h3 id="网站数据统计"><a class="header-anchor" href="#网站数据统计">#</a> 网站数据统计</h3>
<p><img src="https://gitee.com/veal98/images/raw/master/img/20210208170801.png" alt=""></p>
<h3 id="帖子热度计算"><a class="header-anchor" href="#帖子热度计算">#</a> 帖子热度计算</h3>
<p>每次发生点赞（给帖子点赞）、评论（给帖子评论）、加精的时候，就将这些帖子信息存入缓存 Redis 中，然后通过分布式的定时任务 Spring Quartz，每隔一段时间就从缓存中取出这些帖子进行计算分数。</p>
<p>帖子分数/热度计算公式：分数（热度） = 权重 + 发帖距离天数</p>
<!--beforebegin--><div class="language-java extra-class"><!--afterbegin--><pre v-pre class="language-java"><code><span class="token comment">// 计算权重</span>
<span class="token keyword">double</span> w <span class="token operator">=</span> <span class="token punctuation">(</span>wonderful <span class="token operator">?</span> <span class="token number">75</span> <span class="token operator">:</span> <span class="token number">0</span><span class="token punctuation">)</span> <span class="token operator">+</span> commentCount <span class="token operator">*</span> <span class="token number">10</span> <span class="token operator">+</span> likeCount <span class="token operator">*</span> <span class="token number">2</span><span class="token punctuation">;</span>
<span class="token comment">// 分数 = 权重 + 发帖距离天数</span>
<span class="token keyword">double</span> score <span class="token operator">=</span> <span class="token class-name">Math</span><span class="token punctuation">.</span><span class="token function">log10</span><span class="token punctuation">(</span><span class="token class-name">Math</span><span class="token punctuation">.</span><span class="token function">max</span><span class="token punctuation">(</span>w<span class="token punctuation">,</span> <span class="token number">1</span><span class="token punctuation">)</span><span class="token punctuation">)</span>
        <span class="token operator">+</span> <span class="token punctuation">(</span>post<span class="token punctuation">.</span><span class="token function">getCreateTime</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">.</span><span class="token function">getTime</span><span class="token punctuation">(</span><span class="token punctuation">)</span> <span class="token operator">-</span> epoch<span class="token punctuation">.</span><span class="token function">getTime</span><span class="token punctuation">(</span><span class="token punctuation">)</span><span class="token punctuation">)</span> <span class="token operator">/</span> <span class="token punctuation">(</span><span class="token number">1000</span> <span class="token operator">*</span> <span class="token number">3600</span> <span class="token operator">*</span> <span class="token number">24</span><span class="token punctuation">)</span><span class="token punctuation">;</span>
</code></pre>
<!--beforeend--></div><!--afterend--><p><img src="https://gitee.com/veal98/images/raw/master/img/20210208173636.png" alt=""></p>
<h2 id="📖-配套教程"><a class="header-anchor" href="#📖-配套教程">#</a> 📖 配套教程</h2>
<p>想要自己从零开始实现这个项目或者深入理解的小伙伴，可以扫描下方二维码关注公众号『<strong>飞天小牛肉</strong>』，第一时间获取配套教程, 不仅会详细解释本项目涉及的各大技术点，还会汇总相关的常见面试题，目前尚在更新中。</p>
<img src="https://gitee.com/veal98/images/raw/master/img/20210204145531.png" style="zoom:67%;" />
<h2 id="📞-联系我"><a class="header-anchor" href="#📞-联系我">#</a> 📞 联系我</h2>
<p>有什么问题也可以添加我的微信，记得备注来意：格式 <u>（学校或公司 - 姓名或昵称 - 来意）</u></p>
<img width="260px" src="https://gitee.com/veal98/images/raw/master/img/微信图片_20210105121328.jpg" >
<h2 id="👏-鸣谢"><a class="header-anchor" href="#👏-鸣谢">#</a> 👏 鸣谢</h2>
<p>本项目参考<a href="https://www.nowcoder.com/" target="_blank" rel="noopener noreferrer">牛客网<OutboundLink/></a> — Java 高级工程师课程，感谢老师和平台</p>
</ContentSlotsDistributor>
"},null]} \ No newline at end of file diff --git a/node_modules/@vuepress/core/node_modules/.cache/vuepress/cf95a4f56065996d75717bef77e96a78.json b/node_modules/@vuepress/core/node_modules/.cache/vuepress/cf95a4f56065996d75717bef77e96a78.json index 4ae96aac..0511c58d 100644 --- a/node_modules/@vuepress/core/node_modules/.cache/vuepress/cf95a4f56065996d75717bef77e96a78.json +++ b/node_modules/@vuepress/core/node_modules/.cache/vuepress/cf95a4f56065996d75717bef77e96a78.json @@ -1 +1 @@ -{"remainingRequest":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js??ref--1-1!E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js??ref--1-2!E:\\GreateCommunity\\docs\\Guide.md","dependencies":[{"path":"E:\\GreateCommunity\\docs\\Guide.md","mtime":1613049784185},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:aW1wb3J0IHsgcmVuZGVyLCBzdGF0aWNSZW5kZXJGbnMgfSBmcm9tICIuL0d1aWRlLm1kP3Z1ZSZ0eXBlPXRlbXBsYXRlJmlkPTFhNzBkNzA3JiIKdmFyIHNjcmlwdCA9IHt9CgoKLyogbm9ybWFsaXplIGNvbXBvbmVudCAqLwppbXBvcnQgbm9ybWFsaXplciBmcm9tICIhLi4vbm9kZV9tb2R1bGVzL3Z1ZS1sb2FkZXIvbGliL3J1bnRpbWUvY29tcG9uZW50Tm9ybWFsaXplci5qcyIKdmFyIGNvbXBvbmVudCA9IG5vcm1hbGl6ZXIoCiAgc2NyaXB0LAogIHJlbmRlciwKICBzdGF0aWNSZW5kZXJGbnMsCiAgZmFsc2UsCiAgbnVsbCwKICBudWxsLAogIG51bGwKICAKKQoKLyogaG90IHJlbG9hZCAqLwppZiAobW9kdWxlLmhvdCkgewogIHZhciBhcGkgPSByZXF1aXJlKCJFOlxcR3JlYXRlQ29tbXVuaXR5XFxub2RlX21vZHVsZXNcXHZ1ZS1ob3QtcmVsb2FkLWFwaVxcZGlzdFxcaW5kZXguanMiKQogIGFwaS5pbnN0YWxsKHJlcXVpcmUoJ3Z1ZScpKQogIGlmIChhcGkuY29tcGF0aWJsZSkgewogICAgbW9kdWxlLmhvdC5hY2NlcHQoKQogICAgaWYgKCFhcGkuaXNSZWNvcmRlZCgnMWE3MGQ3MDcnKSkgewogICAgICBhcGkuY3JlYXRlUmVjb3JkKCcxYTcwZDcwNycsIGNvbXBvbmVudC5vcHRpb25zKQogICAgfSBlbHNlIHsKICAgICAgYXBpLnJlbG9hZCgnMWE3MGQ3MDcnLCBjb21wb25lbnQub3B0aW9ucykKICAgIH0KICAgIG1vZHVsZS5ob3QuYWNjZXB0KCIuL0d1aWRlLm1kP3Z1ZSZ0eXBlPXRlbXBsYXRlJmlkPTFhNzBkNzA3JiIsIGZ1bmN0aW9uICgpIHsKICAgICAgYXBpLnJlcmVuZGVyKCcxYTcwZDcwNycsIHsKICAgICAgICByZW5kZXI6IHJlbmRlciwKICAgICAgICBzdGF0aWNSZW5kZXJGbnM6IHN0YXRpY1JlbmRlckZucwogICAgICB9KQogICAgfSkKICB9Cn0KY29tcG9uZW50Lm9wdGlvbnMuX19maWxlID0gImRvY3MvR3VpZGUubWQiCmV4cG9ydCBkZWZhdWx0IGNvbXBvbmVudC5leHBvcnRz"}]} \ No newline at end of file +{"remainingRequest":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js??ref--1-1!E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js??ref--1-2!E:\\GreateCommunity\\docs\\Guide.md","dependencies":[{"path":"E:\\GreateCommunity\\docs\\Guide.md","mtime":1613050811891},{"path":"E:\\GreateCommunity\\node_modules\\cache-loader\\dist\\cjs.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\vue-loader\\lib\\index.js","mtime":499162500000},{"path":"E:\\GreateCommunity\\node_modules\\@vuepress\\markdown-loader\\index.js","mtime":499162500000}],"contextDependencies":[],"result":[{"type":"Buffer","data":"base64:aW1wb3J0IHsgcmVuZGVyLCBzdGF0aWNSZW5kZXJGbnMgfSBmcm9tICIuL0d1aWRlLm1kP3Z1ZSZ0eXBlPXRlbXBsYXRlJmlkPTFhNzBkNzA3JiIKdmFyIHNjcmlwdCA9IHt9CgoKLyogbm9ybWFsaXplIGNvbXBvbmVudCAqLwppbXBvcnQgbm9ybWFsaXplciBmcm9tICIhLi4vbm9kZV9tb2R1bGVzL3Z1ZS1sb2FkZXIvbGliL3J1bnRpbWUvY29tcG9uZW50Tm9ybWFsaXplci5qcyIKdmFyIGNvbXBvbmVudCA9IG5vcm1hbGl6ZXIoCiAgc2NyaXB0LAogIHJlbmRlciwKICBzdGF0aWNSZW5kZXJGbnMsCiAgZmFsc2UsCiAgbnVsbCwKICBudWxsLAogIG51bGwKICAKKQoKLyogaG90IHJlbG9hZCAqLwppZiAobW9kdWxlLmhvdCkgewogIHZhciBhcGkgPSByZXF1aXJlKCJFOlxcR3JlYXRlQ29tbXVuaXR5XFxub2RlX21vZHVsZXNcXHZ1ZS1ob3QtcmVsb2FkLWFwaVxcZGlzdFxcaW5kZXguanMiKQogIGFwaS5pbnN0YWxsKHJlcXVpcmUoJ3Z1ZScpKQogIGlmIChhcGkuY29tcGF0aWJsZSkgewogICAgbW9kdWxlLmhvdC5hY2NlcHQoKQogICAgaWYgKCFhcGkuaXNSZWNvcmRlZCgnMWE3MGQ3MDcnKSkgewogICAgICBhcGkuY3JlYXRlUmVjb3JkKCcxYTcwZDcwNycsIGNvbXBvbmVudC5vcHRpb25zKQogICAgfSBlbHNlIHsKICAgICAgYXBpLnJlbG9hZCgnMWE3MGQ3MDcnLCBjb21wb25lbnQub3B0aW9ucykKICAgIH0KICAgIG1vZHVsZS5ob3QuYWNjZXB0KCIuL0d1aWRlLm1kP3Z1ZSZ0eXBlPXRlbXBsYXRlJmlkPTFhNzBkNzA3JiIsIGZ1bmN0aW9uICgpIHsKICAgICAgYXBpLnJlcmVuZGVyKCcxYTcwZDcwNycsIHsKICAgICAgICByZW5kZXI6IHJlbmRlciwKICAgICAgICBzdGF0aWNSZW5kZXJGbnM6IHN0YXRpY1JlbmRlckZucwogICAgICB9KQogICAgfSkKICB9Cn0KY29tcG9uZW50Lm9wdGlvbnMuX19maWxlID0gImRvY3MvR3VpZGUubWQiCmV4cG9ydCBkZWZhdWx0IGNvbXBvbmVudC5leHBvcnRz"}]} \ No newline at end of file