背景痛点:毕设开发的三座大山
每年 3 月,实验室里最常听到的三句话是:“依赖又冲突了”“接口怎么又 500 了”“服务器咋连不上”。把这些问题拆开看,其实就是三座大山:
- 重复劳动:登录、分页、文件上传,每换一个新模块都要重新 Ctrl+C/V,代码行数飙升,逻辑却原地踏步。
- 环境黑洞:本地 MySQL 8.0,服务器 5.7;JDK 版本从 11 到 17 来回横跳,Maven 依赖冲突能把 IDEA 的内存直接打满。
- 部署马拉松:手动打 jar→微信传文件→SSH 连服务器→kill 旧进程→nohup 启动,一套组合拳下来,答辩 PPT 都没时间做。
时间只有 8 周,需求却一周三变,效率低意味着直接“延毕”。想破局,得先给项目装上一套“外挂”。
技术选型:为什么不是 SSM 而是 Spring Boot
大三下学期做过一次 SSM(Spring + SpringMVC + MyBatis)课设,配置文件写到怀疑人生:applicationContext.xml、spring-mvc.xml、web.xml 叠成三层汉堡,光扫包路径就能打错三次。相比之下,Spring Boot 把“约定大于配置”写进 DNA,开箱即用的 parent 与 starter 让依赖管理从 50 行 xml 缩成 5 行 gradle。下面这张对比图可以一眼看出差距:
辅助工具也顺势升级:
- Lombok:一键消除 70% 的 getter/setter/toString,实体类瞬间瘦身。
- MyBatis-Plus:内置通用 Mapper 与 IService,单表接口 0 SQL。
- MapStruct:编译期生成 Bean 转换,比 BeanUtils 省 90% 运行耗时。
三者叠加,平均一个模块代码量从 800 行降到 200 行以内,编译速度提升 40%,留给写业务的时间肉眼可见地变多。
核心实现:搭好可复用的“积木”
1. 项目骨架——Spring Initializr 三件套
打开 https://start.spring.io,选 Spring Boot 3.2 + Java 17 + Gradle,依赖勾 Lombok、Spring Web、MyBatis Framework、Validation、Spring Security,30 秒生成 zip。解压后先把包名改成com.grad.xxx,防止后续踩坑。
2. 模块化分包——“纵向切”+“横向切”
纵向按业务:user、file、notice;横向按层次:common、service、dao、web。这样 controller 只依赖 service,service 只依赖 dao,层次清晰,IDEA 的 Dependency Matrix 一片绿色。
3. 通用 CRUD——BaseService 模板
public interface BaseService<T> { IService<T> getRepository(); // MyBatis-Plus 内置 default T findById(Serializable id){ return getRepository().getById(id); } default boolean saveEntity(T entity){ return getRepository().save(entity); } // 更多通用方法... }业务 Service 只需继承并指定实体,单表接口瞬间拥有。新增字段时,0 改动即可同步。
4. 统一响应——ResultVO + GlobalExceptionHandler
@Data @AllArgsConstructor public class ResultVO<T> { private Integer code; private String msg; private T data; public static <T> ResultVO<T> ok(T data){ return new ResultVO<>(200, "success", data); } }配合@RestControllerAdvice捕获所有异常,前端同学再也不用猜 200 里藏没藏错误。
5. DTO 转换——MapStruct 一步到位
@Mapper(componentModel = "spring") public interface UserConverter { UserConverter INSTANCE = Mappers.getMapper(UserConverter.class); UserDTO toDto(User po); }编译期生成实现类,运行期零反射,性能比 Apache BeanUtils 高 10 倍。
自动化部署:push 代码即上线
手动部署三步曲:打包→上传→重启,最容易在“上传”环节断网重来。用 GitHub Actions 做成流水线,代码推送即触发,全程无人工:
- 在项目根目录新建
.github/workflows/deploy.yml:
name: CI-CD on: push: branches: [ main ] jobs: build: runs-on: ubuntu-latest steps: - uses: actions/checkout@v4 - uses: actions/setup-java@v3 with: distribution: 'temurin' java-version: '17' - run: ./gradlew bootJar - name: scp jar to server uses: appleboy/scp-action@v0 with: host: ${{ secrets.HOST }} username: ${{ secrets.USER }} key: ${{ secrets.SSH_KEY }} source: build/libs/app.jar target: /opt/grad - name: restart app uses: appleboy/ssh-action@v1 with: host: ${{ secrets.HOST }} username: ${{ secrets.USER }} key: ${{ secrets.SSH_KEY }} script: | systemctl restart grad-app- 在 GitHub 仓库 Settings → Secrets 里把服务器 IP、账号、私钥一次性配好,以后每次 push 都能看到绿色小勾,3 分钟之内最新代码已在云服务器奔跑。
性能与安全:给生产加两道锁
- 连接池:默认 HikariCP,并发 200 以内无需调参;如需更大并发,把
maximum-pool-size提到 CPU*2+1 即可。 - 日志脱敏:统一用 Logback + PatternLayout,把手机号、身份证配置正则替换为
*********,防止控制台打印泄露隐私。 - 容器内存:2C4G 服务器跑 FatJar 默认占 600 M,加
-Xms512m -Xmx1g限制,留 1 G 给 MySQL 与 Redis,稳稳够用。
避坑指南:前人踩过的坑,后人别再跳
- 事务失效:Spring Boot 默认用 CGLIB,方法必须是 public 且同类调用才生效;写
@Transactional时别图省事放私有方法。 - 热部署不生效:IDEA 2023 以后把
spring-boot-devtools标记为可选依赖,需要手动在 Settings → Build → Compiler 打开 “Build project automatically”。 - 打包路径错误:Gradle 打出的 jar 在
build/libs,别傻乎乎去build/classes找;脚本里写错一次,Actions 日志会疯狂报 “No such file”。 - 多模块循环依赖:common 模块被所有模块依赖,千万别反向引用 web 模块,否则 Maven 直接给你 CircularDependency 大红叉。
- 时区错乱:服务器默认 UTC,数据库时间戳差 8 小时;在
application.yml里加spring.jackson.time-zone=Asia/Shanghai统一收口。
写在最后:把省出来的时间用在创意上
做完这套流程,我的毕设“校园二手书交易平台”代码量从预估 1.2 万行压到 4 千行,功能点却多出“即时聊天”与“扫码收书”。省下来的时间,我把算法推荐章节写得比开题报告还丰满,答辩老师直接问“有没有兴趣继续投期刊”。效率提升不是目的,而是把最宝贵的青春留给真正想探索的方向。
如果你也在毕设泥潭里挣扎,不妨按本文顺序,先把骨架搭起来,再把业务一块块填进去。push 一次就能看到服务器跑通的感觉,会上瘾。动手重构你的项目吧,下一个绿色小勾就在前方。