关于我是如何从零自己设计、全栈编写出个人小站的经历
这篇文章是关于我第一次个人的全栈项目 - littleSharing 的开发经历的一篇文章,主要讲述了我是如何从零自己设计、全栈编写出个人小站的。
前言
其实本来很早就开始打算为我的个人项目稍微的引个流了,不过由于之前一直在进行功能更新以及 bug 修复的工作,所以一直打算等把整个项目开发的能够使用了之后再写个文稍微介绍一下。就在前几天把文章资源的上传全部转移至腾讯云 COS 了之后,我才打算动笔写一写这段经历,并给出 github
仓库地址以及访问的网址,希望可以给想要学习前后端的大家一点参考。
这个项目本来是打算当成我的个人博客来用的,不过转念一想似乎这个项目其实已经完全可以算成一个个人实现的作品了,集成的功能之后也会随着更新逐渐增多,因此美其名曰 “知识分享小站” 。
前端代码地址:https://github.com/nonhana/littleSharing-Frontend
后端代码地址:https://github.com/nonhana/littleSharing-Backend-TS
这个项目是完完全全由我 100%从 0 开始进行开发到上线,包括:模块设计、功能设计、前端界面 UI 设计、数据库设计、后端接口设计、前端代码编写、后端代码编写、联调测试、部署上线等等。 我写这篇文章的目的也是为了分享一下我的这段经历。
先讲一下这个项目的开发背景。这个项目项目创立之初的目标有两个:
- 记录自己在开发过程中所遇到的各种困难,作为自己的积累之地;
- 为了巩固、实践自己所学到的最前沿的技术到实际运用上面,并在不断的 bug 调试当中优化自己的编程思想。
综上,我会从 项目设计 、 技术实现 两个方面来讲述这个项目。先贴几张项目截图:
二、项目设计
对于一个项目,奠定了整个开发基调的就是位于最开始的项目设计。
按照我自己的经验,我将一个新项目的设计步骤划分为:功能模块划分、前端 UI 界面设计、数据库设计、API 接口设计。
其实本来还得在功能模块划分之前加一个用户需求分析的步骤,不过对于我自己来说,一个项目的开发更多的是基于我本身的需求或者说是基于兴趣,因此没有必要特地的进行用户需求分析。
功能模块划分
在项目开发设计的时候,功能模块的划分往往意味着要编写 哪几类 的页面。
这个项目在最新的版本,总共可以划分为如下几个功能模块:
登录页
登录页面,顾名思义就是用户第一次访问或者登录失效的时候会跳转到的页面。在这个页面主要就是提供一个欢迎页,并且提醒进入的用户先进行登录或者注册的操作。当注册成功后,窗口会变为登录窗口;当注册成功后,会跳转到首页。
首页
首页是本站最核心的内容之一,因此需要展现的内容也是较为丰富的。首页主要展现的有:文章列表、文章分类搜索栏、文章浏览趋势、最新发布的五篇文章列表以及顶部导航栏。
其中文章列表是按照用户的浏览趋势进行推荐获取的。文章列表由一个个文章项目组成,包含了文章除了主要内容以外的全部基本信息,并且包括了文章作者的一系列信息。用户可以通过点击项目标题、简介进入文章详情页面;点击用户头像进入用户个人主页。
文章分类搜索栏是为了给用户筛选出或者搜索出自己想要的文章而编写的。搜索栏提供了很多学科的分类,并且也提供了文章搜索关键字来进行检索。检索完毕之后,可以点击“还原”按钮进行还原。
文章浏览趋势是用于记录用户浏览具有某类标签的文章的次数的。当用户点击进入某篇文章的详情页的时候,会自动地把标签提交给后端进行记录,并且根据浏览时间将其划分到第几个月份,最终在折线图上进行趋势展示。
最新发布的五篇文章列表就是获取到按时间顺序倒序的最新的文章列表。
顶部导航栏提供了最左侧的页面 logo 按钮,点击就可以返回主页;中间的搜索框,目前暂时只作为装饰使用;右侧的消息中心按钮、上传文章按钮,点击可以跳转到对应的页面;最右侧的用户个人头像,鼠标移到头像处会显示下拉框,可以选择退出登录还是进入可个人主页。顶部导航栏几乎在任何页面都存在,除了 登录页 和 消息中心页面 。
文章详情页
文章详情页是用户浏览文章的主要页面。文章详情页主要的作用为浏览文章、记录文章书签、文章操作(点赞、收藏、评论、转发)等。另外也提供了相似文章推荐、作者信息栏展示等功能,便于用户去浏览其他的文章或者前往作者主页。
文章编写、上传页
文章编写、上传页可以通过顶部导航栏的上传文章按钮进入。在这个页面,文章的编写分为两步:1. 文章基本信息的填写;2. 文章内容的填写。文章基本信息又分为转载文章与非转载文章,区别在于是否需要填入原文地址。文章基本信息填写完成之后,点击下一步可以进入到文章内容的填写,采用 Markdown 的文章格式。
消息中心页
消息的作用是为了提醒用户之间的互动行为,以及官方的通知送达。
按照基本的分析,可以把消息作如下分类:
- 用户消息
- 被点赞、收藏消息(文章和评论可共用,也就是评论被点赞、文章被点赞是一样的)
- 评论消息(自己的文章被评论、自己的评论被回复)
- 被其他用户关注消息
- 关注的用户发布新文章消息
- 系统消息
- 文章审核通过与否(目前并未配置后台审核,因此暂时没有用上,后续准备弄个模板)
为了使消息能够切实的被用户注意到并且有意的去查看,需要引入 未读标记(badge) 的形式来提醒用户自己有未读消息需要处理。
个人中心页
个人中心给用户浏览在本站记录的与自己相关的数据,包括:自己发布的文章列表、自己收藏的文章列表、自己的关注/粉丝列表、自己的数据、自己的资料。
发布的文章列表也就是用户自己在这个站内发布的文章列表,以专门的文章项目进行展示,并且可以点击右上角确定是否删除或者重新编辑。
收藏的文章列表是用户在浏览文章的过程中进行的收藏操作,会将该文章添加进用户的收藏列表,之后可以在这里访问到收藏的文章。
自己的关注/粉丝列表能够获取到用户自己的关注列表或者被关注的用户列表进行展示,显示了一些用户的基本信息,点击可以进入其他用户的个人主页。
自己的数据记录了用户在本站所获得的一系列数据。包括:浏览文章关键词统计饼图、发布文章类别统计饼图、获得的点赞量变化曲线、获得的收藏量的变化曲线。
自己的资料就是用户自己设置自己相关的信息的地方。用户可以在这个页面浏览、修改自己的个人基本信息,包括用户名、头像、背景图等等信息。 如果是浏览他人的个人主页,无法进入到他人的个人资料页。
前端 UI 界面设计
基于上述的功能模块分析,基本上要设计哪几个页面、页面中所出现的元素如何进行有效的组合,应该也是一目了然了。在 UI 设计中,我采用了 MasterGO
这一款原型设计软件进行界面样式的设计。以下贴一些当初的设计截图:
其实看得出来,现在的版本和最初的 UI 设计相差的有点大,不过整体的实现基调是没有发生太大变化的。
我认为一个好的 UI 设计,重要的不是每个页面具体的样式是如何设计的,而是整个项目的整体风格的确定。 只有当整个项目风格确定了之后,才能够有条不紊的推进每个具体页面、具体细节的设计制作。比如我在另一个项目中在整体设计开始之前的公共约定:
有了这个约定之后,整个项目的设计都基于一个固定的范式,无论是对于开发者来说还是设计者来说都能够基于这一套模板进行样式的编写与设计,能够极大的提高开发的效率。 其实更多的是给整个项目一个风格基调的确定吧。。。不然每个页面都有着自己的风格设计,我觉得这才是最致命的一点(因为我就吃过这个亏)。
数据库设计
我是在 UI 设计基本完成之后再进行数据库表的设计的。
在数据库表的设计过程中,我并没有特别的去注重所谓的设计范式(Normalization),而是着眼于这个项目中所包含哪几个重要的 数据实体(entity) ,先将最基本的实体类型划分好,然后再在这个实体的基础之上根据需要实现的功能或者这个实体包含另外的同样也需要另外的实体去表示的数据,那么再加二级实体、关联表或者外键,形成一对多、多对多、一对一的关系。
实体划分:“一级实体”、“二级实体”
那么数据实体该怎么去划分呢? 我们首先应该清楚一点,一个数据实体必须要包含最基本的几个要素:增删改查,并且自身需要包含某种程度的信息。 但凡某个对象是涉及到增删改查四种类型的操作,并且有着自己的属性信息的,都是可以划分为一个实体出来的。
在基本实体的定义中,可以根据我们的原型设计图与功能模块分析去做一个初步的推断。比如我在功能模块划分中着重提到了文章、用户、评论等几个概念,我们就可以先定下一级实体:文章(articles)、用户(users)、评论(comments),然后我们可以基于这最基本的实体,分析它们当中又包含哪些信息是需要其他的实体去表示的,也就是二级实体。
实体和实体之间的关联关系:一对多、一对一、多对多
一对多
我们以用户(users)这个基本实体举例,一个用户基本信息包括用户头像、用户名、用户签名、用户浏览文章关键字等等。而在我们的设计中,用户关键字是在用户浏览文章的过程中自己进行添加的,并且具有自己有意义的字段:关键字浏览次数(keywords_count)、关键字名称(keywords_name),因此有必要将其单独提取出一个数据实体:关键字(keywords)。由于一个用户可以有多个浏览关键字,一个浏览关键字的相关数据只能属于一个用户,因此用户与关键字之间是一对多的关系,其外键需要加载 keywords 这个数据实体表当中,和 users 表相关联。
如果一级实体之前也存在关联关系,那么思路也是和之前一样进行分析。比如一篇文章只有一个作者,而一个作者可以发很多篇的文章,这是一个一对多的关系,需要在 articles 表当中加一个 author_id 字段来标明作者的 id,和 users 表相关联。
不过也并非一切一对多关联的实体都需要通过外键来约束。 在特定的情况下,比如某个实体的内容对于另外一个实体来说是 共用的 ,那就没必要通过外键来进行约束,直接在另外的那一个实体内部进行 id 的添加即可。比如对于文章标签(article_labels)这一个实体,我们知道一个文章是可以加多个标签的,本来应该是一对多的关系,外键加在 article_labels 这个表中是第一直觉;但是实际上,大多数的文章都会重复的使用同一个文章标签来标明自己的所属类别,意味着如果采用了外键的形式,那么数据库表中大多数 label_name 字段都会是重复的。并且对于前端来说,文章标签是用户自行选择添加的,主要方式并不是自己输入,因此获取到不重复的文章标签列表也是重要的功能。基于以上几点的考虑,文章标签是没必要通过外键关联 articles 表,而是直接在 articles 表中加一个 article_labels 字段,其中包含了以英文逗号为分隔的 label_id 列表来标明这篇文章的标签,这样子才能够确保文章标签表的干净,也便于前后端的接口实现。
总结一下,面对一个一对多的关系:
- 如果“多”的这个实体所包含的信息对于每个“一”的实体来说 是唯一的 ,那么有必要在“多”的实体表中加一个关联“一”表的外键;
- 如果“多”的这个实体所包含的信息对于每个“一”的实体来说 是共用的 ,那么只需要在“一”的实体表中加一个包含“多”的主键字段,并且采用字符串拼接的方式存储。
不过面对第二种情况,也就是“多”的实体对于每个“一”的实体是共用的时,我事后咨询了一下 GPT,他提示我说更加遵循标准的做法使用一个额外的关联表(或称为连接表、交叉表),该表存储“一”和“多”实体间的关系。
通常情况下,不推荐在“一”的实体表中通过字符串拼接方式存储“多”的主键。这种做法虽然看似简单,但会导致几个问题:
- 查询效率低:如果需要查询所有与某个“多”实体关联的“一”实体,需要解析每个“一”实体中的这个字段,这在大型数据库中效率非常低。
- 数据完整性和一致性难以维护:如果“多”实体的某些信息改变,需要更新所有相关联的“一”实体记录,这容易出错且效率低下。
- 违反数据库范式原则:这种设计通常违反了第一范式(1NF),因为字段值不是原子的。
例如,如果有一个学生表和一个课程表,一个课程可能由多个学生共享,可以创建一个额外的表来存储学生 ID 和课程 ID 的关系。
所以可能,我事后还得去修改一下我的数据库结构设计。。
多对多
再比如对于用户点赞操作,这个是完全属于功能类的表添加,其中不包含任何具体的信息,仅仅只是提供了关联了用户与文章两者的信息。因此点赞功能的数据库实现是不涉及实体的,涉及到的是中间表的操作。由于一篇文章能被多个用户点赞,一个用户能够点赞多篇文章,是一个多对多的关系,因此,点赞表的设计可以仅设置两个外键:article_id 和 user_id,分别关联 articles 表和 users 表。
一对一
如果两个实体之间是一对一的关系,那么关联的外键可以根据自己的需求,选择加到哪一方。无论加到哪一方都可以正常的查询。
数据库模型
好的,说了这么多,对于我自己设计数据库表的思路也大概就是这样,还是相对比较清晰简单的。接下来我贴出我的 MySQL 数据库表的设计图:
可见这个项目相对比较单纯,涉及到的数据表较少。其中不乏两张孤立的、不和任何表关联的表,这几个表都是起到存储公用数据的作用的。
API 接口设计
数据库设计完成之后,便可以根据需求着手开始设计 API 接口了。
在整体的设计过程中,我也是采用了着眼于 “数据实体” 的方法,对每一个实体需要实现的功能都进行相应的归类,并且以一个接口实现一个特定目的为基准进行设计。
总的而言,在这个项目中接口主要分四类: 用户接口 、 文章接口 、 评论接口 以及 消息接口 。
具体每个接口的设计在这里不再详细的赘述(因为如果细讲的话估计还得专门写篇文章单独讲清楚),详情可访问下方的 Apifox
接口文档地址 ↓
API 接口设计文档地址:https://littlesharing.apifox.cn
三、技术实现
在这个部分,我会着重于当前最新版本的技术栈来进行对项目本身目录结构的讲解,详情部分如果想要找寻参考,可以直接参考源代码。
前端方面
最新版本的前端技术栈使用的是:Vue3 + TypeScript + Pinia + ElementPlus + Vite。
- 环境配置文件 (.env, .env.development, .env.production): 这些文件用于配置不同环境(开发、生产等)的环境变量。
.env
是公用的环境变量配置,.env.devlopment
是开发环境变量配置,.env.production
是部署环境变量配置。 - 配置文件 (.eslintrc.cjs, .prettierrc.cjs, .stylelintrc.cjs): 分别用于配置 ESLint、Prettier 和 StyleLint,这些是代码质量和格式化的工具。
- TypeScript 定义文件 (auto-imports.d.ts, components.d.ts, env.d.ts, mojs.d.ts, vite-env.d.ts): 用于定义 TypeScript 类型。前两个是 element-plus 组件库的自动导入插件自动生成的,
mojs.d.ts
是给mojs
这个动画库临时写的一个声明文件,因为这个库目前是还没有适配 TS 版本,得自己手动临时写一个。 - 入口文件 (index.html): Vue 应用的 HTML 入口文件,里面配了一些公有的样式,比如滚动条。
- 项目配置文件 (package.json, pnpm-lock.yaml, tsconfig.json, tsconfig.node.json, vercel.json, vite.config.ts): 包括包管理、TypeScript 配置、部署配置和 Vite 构建工具配置。这个项目采用的包管理工具为
pnpm
,部署的话是直接放到vercel
上面的,然后通过cloudflare
服务重定向到我自己买的域名,从而使得其可以在国内访问。vercel.json
是配置vercel
部署的文件,防止刷新后页面丢失。 - public 目录: 包含静态资源。
- README.md: 项目的文档说明。
- src 目录: 包含 Vue 应用的主要源代码。
- api 目录: 包含与后端 API 交互的逻辑,分为不同模块,也就是文章、评论、消息、用户。并且每个模块为单独的一个目录,每个目录配有 types 目录,其中的 Index.ts 专门用于接口的类型定义,包括请求参数与返回数据体类型定义。
- assets 目录: 存放静态资源,主要是 SVG 文件。
- components 目录: 包含 Vue 组件,按功能分类(如全局组件、小组件、各种模块组件)。
- main.ts: Vue 应用的主入口文件。
- router 目录: 包含 Vue Router 的配置。
- service 目录: 包含应用服务逻辑,如 HTTP 请求。
- store 目录: 包含 Pinia 状态管理的相关文件。编写方式统一为 modules 下方根据不同的用途进行分类,通过最外侧的 index.ts 统一暴露。
- styles 目录: 包含全局样式文件。主要是修改 element-plus 的主题色的 scss 文件。
- types 目录: 存放自定义的 TypeScript 类型定义。
- utils 目录: 包含工具函数和一些常用常量定义。
- views 目录: 具体的按照页面类别来分的主要页面组件。
后端方面
最新版本的后端技术栈使用的是:原生 Express + TypeScript + MySQL。
- 配置文件 (.eslintrc.cjs, .gitignore, tsconfig.json): 分别用于配置 ESLint(代码质量工具)、Git 忽略文件和 TypeScript 编译选项。最后将这个项目通过 pm2 来部署的时候就是通过 tsconfig.json 来编译为 js 文件后再部署的。
- 入口文件 (app.ts, bin/server.ts):
app.ts
是应用的主入口,定义 Express 应用的基本设置;bin/server.ts
用于启动服务器。 - 常量文件 (constant.ts): 定义整个应用使用的常量。
- controller 目录: 包含处理不同数据模型(如文章、评论、消息、用户)的控制器,通常负责处理请求和返回响应。
- actions.ts: 用于定义具体的业务逻辑操作,通常是这个实体的除了增删改查以外的衍生操作。
- basic.ts: 包含最基本的逻辑,也就是 增删改查 。
- otherData.ts: 用于处理与主要数据模型相关但不是核心部分的数据。
- types/index.ts: 为每个控制器定义 TypeScript 类型。
- 静态文件复制脚本 (copyStatic.ts): 用于复制静态资源。
- 数据库目录 (database): 包含数据库配置(db.config.ts)和数据库连接逻辑(index.ts)。
- 中间件目录 (middleware): 包含中间件,如文件上传(upload.middleware.ts)和用户验证(user.middleware.ts)。
- package.json: 定义项目的依赖和脚本。
- README.md: 项目文档说明。
- 路由目录 (routes): 包含定义 API 路由的文件,路由通常指向不同的控制器。
- 类型定义文件 (types/cos-nodejs-sdk-v5.d.ts): 腾讯云 COS Node.js SDK 的 TypeScript 类型定义。
- 工具函数目录 (utils): 包含一些辅助函数。
- 视图目录 (views): 包含 Jade(现称 Pug)模板文件,用于生成一些特定的 HTML 页面,比如有时候用户直接通过网页 url 来访问后端接口,如果找不到就会直接在网页上显示 404 的错误信息,这就是由这个 Jade 来生成的。
部署方面
这个项目的部署是分开来的,前端是通过 vercel
来进行部署;后端是通过我自己的服务器的 pm2 Node.js 项目管理器来部署。
前端部署
前端的部署相对简单,直接在 vercel
的官网中登录自己的 github 账号,然后选择自己的项目仓库并选择对应的分支之后,配置好命令就可以一键部署了。可以直接一下子帮你做到 CI/CD,是十分方便的一款部署工具,非常适合个人的小项目。
部署了之后,我也将其域名转为国内的自己的域名,能够使其在国内进行访问。具体的将 vercel
部署应用变为在国内可访问的步骤,可以参考一下我之前做记录的文章:如何在国内访问 vercel 部署应用?
除了部署本身,部署这一行为还涉及到几个比较重量级的方面:打包优化以及 SEO(搜索引擎优化)。
- 关于打包优化,主要是借助了
viteCompression
插件对项目资源进行 Gzip 压缩、使用viteImagemin
插件对图片资源大小进行优化、使用 vite 内部的 build 选项来优化模块兼容性和实现代码分割。具体的代码可以去参看源代码。 - 关于 SEO,这个项目采用了
vite-ssg
这款插件进行 SSG 打包,将原本的单页面通过读取vue-router
配置来生成多页面。具体的配置方法可以参考我之前写的一篇文章:怎么在使用 Vue3+Pinia+Vite+TS+ElementPlus 的应用中引入 Vite-SSG 打包,也可以直接对着我的代码进行参考。
后端部署
后端部署使用的是 pm2 这个 Node.js 项目管理工具,部署在我自己的 AWS EC2 实例上。
由于在 pm2 上面跑的 Node.js 项目只支持 JavaScript,而我的 Express 应用是基于 TypeScript 的,因此在部署的时候就避免不了编译。具体的编译选项已经在 tsconfig.json 中开启,输出编译内容到根目录的 dist 文件夹下面。
同时,由于 .env
环境变量文件、静态资源存储目录 public
及其子目录的存在,如果这两个地方发生了变更,也需要亲自手动在服务端修改服务端需要的对应内容。比如环境变量可能本地开发和部署环境是不同的,那么得在远程终端中进行相应的修改。而 public
目录,由于涉及到将文件保存在本地之后再进行 COS 上传的操作,因此对应的路径目录还是需要和本地开发保持一致。具体的结构如下图所示:
所以一般在本地进行开发完毕,到服务端进行部署的流程一般为:
- 本地开发完毕,提交变更到远程仓库;
- 登录远程服务器,在后端项目的存放目录中将新代码进行拉取,如果涉及到新的 npm 包的变更,需要重新进行 install。
- 如果本地开发修改了涉及到
.gitignore
的部分,也就是上面所说的环境变量和静态文件存储目录等,需要在远程终端处进行对应的修改。 - 上面几步进行好后,在根目录处运行
npm run build
,进行 ts 转 js 的打包。最终打包完成后,会输出到 dist 目录。 - 如果之前已经有启动项目,需要先运行
pm2 stop XXX
将项目停掉。确认项目未启动后,运行pm2 start XXX pm2 start ./dist/bin/server.js
。这边的XXX
就是这个后端项目的名称,你自行命名。运行完毕后,可以通过pm2 list
来查看 pm2 正在管理的 Node.js 项目。
然后,给出本地开发的 .env
文件的变量配置,各位如果想要自己在本地尝试运行的话可以参考:
最后,我的这个 Express 后端应用是配置了 SSH 证书的,也就是可以通过 https+域名的方式进行后端接口的访问。这点是通过环境变量的 NODE_ENV
来确定的,如果是本地开发就不启用证书,在远程部署则调用存放在本地的 SSH 证书进行部署。具体的在 Express 项目上部署 SSH 证书的步骤,有需要的同学可以自行去搜索一下,这边我简单贴一下代码截图:
五、结语
以上就是全部的关于 littleSharing
这个项目的介绍了,算是从最最最开始的设计到实现一步步说明过来,总结了我自己的一些开发经验。当然经验肯定不是一蹴而就的,也是经过了相对比较多的版本迭代了之后才慢慢总结出来的一些东西。
之后关于这个项目,也是想着进行前后端两方面的重构与优化。
- 前端部分可能总体的大框架部分就不会再变更了,也就是技术栈不会发生太大的变更,不过之后会着重于移动端适配以及其他的一些代码部分优化。
- 后端部分考虑从原本的 Express 框架换成 Nest.js 框架。后者也是我最近在学习的一款非常优秀的 Node.js 框架,提供了非常完整清晰的开发目录以及流程。原生的 Express 虽然轻量,但同时也导致了其不易维护,难以实现标准化的构建,也就是最佳实践。
最近也是在打算开一个新坑: Picals
,一个模仿 Pixiv
的插画收藏网站,使用 Nuxt3 + Nest.js 作为主要技术栈。可以看我的前几篇文章,基本都是用的我之前存的图图。由于我本人是一个存图怪,又有比较强的收藏欲,因此老早就想有这么一个属于自己的存图空间了,因此开始计划了这么一个项目。如果开发进行了,我也会专门写点东西来介绍一下开发的经验过程的。
那就先这样吧,感谢你能够读到这里。如果对这个项目感到满意或者受到了一些小小的借鉴启发,希望能够点个小小的 star!~