从 Struts2 单体到 Spring Cloud 微服务:一个 P2P 系统的真实重构之路(2019 年实战复盘)

从 Struts2 单体到 Spring Cloud 微服务:一个 P2P 系统的真实重构之路(2019 年实战复盘)

开源地址https://gitee.com/lijinquanBlog/lijinquan-p2p
关键词:微服务重构、Spring Cloud、Eureka、Feign、Hystrix、P2P 系统、架构演进


🌟 写在前面:这不是 Demo,而是一次"生存式"重构

2019 年初,我手头有一个运行多年的 P2P 投资平台------基于 Struts2 + Spring + MyBatis + Oracle 的典型单体架构。随着用户增长,系统暴露出严重问题:

  • 每次发版需全量部署,上线风险高
  • 团队协作困难:前端改页面,后端动接口,测试全回归
  • 高峰期数据库连接池打满,投资下单经常超时
  • 权限、账户、债权逻辑耦合,改一处崩三处

于是,我决定:彻底重构为微服务架构

没有大厂资源,没有团队支持,只有一个人、一台 Dell 笔记本,和一个信念:"能跑起来,就能活下去。"


🔧 一、为什么选择 Eureka + Feign + Hystrix?(不是没得选,而是选得稳)

很多人问我:"2019 年都快 Nacos 了,为啥不用?"

答案很简单:时间点不对。

  • Nacos :2018 年 7 月才开源,2019 年初仍是 0.x 版本,无生产级案例,文档匮乏。
  • Sentinel :虽已开源,但 Spring Cloud 集成不成熟,连自动装配都靠手写
  • Eureka + Feign + Hystrix :Netflix 套件,经过 Netflix、国内电商多年验证,社区问题一搜就有解。

💡 在资金敏感的 P2P 系统里,稳定性 > 新潮。

我选择了"能让我睡着觉"的技术栈。


🏗️ 二、如何拆分服务?从业务出发,而非技术炫技

我没有盲目按"用户、订单、商品"拆,而是根据 P2P 核心业务域

微服务 职责 端口
microservicecloud-provider-user-8001 用户注册、登录、实名认证 8001
microservicecloud-provider-invest-8002 投资、债权匹配、收益计算 8002
microservicecloud-provider-account-8003 资金账户、充值提现(对接支付) 8003
microservicecloud-consumer-web-81 前台 Web(Angular)调用聚合层 81
microservicecloud-eureka-7001 服务注册中心 7001

关键原则

  • 每个服务独立数据库(避免跨库事务)
  • 异步解耦:投资成功后,通过 ActiveMQ 通知账户系统扣款
  • 统一入口:Web 层通过 Feign 调用多个 Provider,前端无感知

⚙️ 三、踩过的坑与解决方案(全是血泪经验)

❌ 坑 1:Feign 调用默认超时 1 秒,投资下单直接失败

复制代码

yaml

复制代码
# 解决方案:显式配置超时
feign:
  client:
    config:
      default:
        connectTimeout: 5000
        readTimeout: 10000

❌ 坑 2:Redis Token 多服务共享,登出失效难

  • 方案:Token 存 Redis,Key = user:token:{userId}
  • 登出时删除该 Key,所有服务读取同一 Redis 实例

❌ 坑 3:Hystrix 熔断后,用户看到"系统繁忙",体验差

  • 优化:自定义 fallback 返回友好提示,并记录日志供运维分析

📊 四、效果如何?数据不会说谎

指标 重构前(单体) 重构后(微服务)
部署时间 15 分钟(全量) 3 分钟(单服务)
并发能力 ≈ 50 TPS ≈ 500 TPS(本地压测)
故障隔离 一处崩,全站挂 投资服务挂,用户仍可查账户
开发效率 全员阻塞 前端 + 后端并行开发

💡 虽然没上云、没做 CI/CD,但在资源有限的个人项目中,这已是质的飞跃


❤️ 五、给后来者的建议

  1. 不要为了微服务而微服务:先问"业务是否需要拆?"
  2. 技术选型看成熟度,不看热度:2019 年用 Eureka,是专业;硬上 Nacos,是冒险。
  3. 文档比代码更重要:我在 README 里写了架构图、启动顺序、依赖关系------这是项目能被理解的关键。
  4. 开源不是为了 Star,而是为了证明"我能搞定复杂系统"
相关推荐
NE_STOP1 小时前
MyBatis-配置文件解读及MyBatis为何不用编写Mapper接口的实现类
java
后端AI实验室6 小时前
用AI写代码,我差点把漏洞发上线:血泪总结的10个教训
java·ai
程序员清风8 小时前
小红书二面:Spring Boot的单例模式是如何实现的?
java·后端·面试
belhomme8 小时前
(面试题)Redis实现 IP 维度滑动窗口限流实践
java·面试
Be_Better8 小时前
学会与虚拟机对话---ASM
java
开源之眼10 小时前
《github star 加星 Taimili.com 艾米莉 》为什么Java里面,Service 层不直接返回 Result 对象?
java·后端·github
Maori31611 小时前
放弃 SDKMAN!在 Garuda Linux + Fish 环境下的优雅 Java 管理指南
java
用户9083246027311 小时前
Spring AI 1.1.2 + Neo4j:用知识图谱增强 RAG 检索(上篇:图谱构建)
java·spring boot
小王和八蛋12 小时前
DecimalFormat 与 BigDecimal
java·后端
beata12 小时前
Java基础-16:Java内置锁的四种状态及其转换机制详解-从无锁到重量级锁的进化与优化指南
java·后端