单体架构:Spring Boot

    如果你此时并不曾了解过什么是“The Fenix Project”,建议先阅读这部分内容

    单体架构是 Fenix’s Bookstore 服务端的起始版本,它与此后基于微服务(Spring Cloud、Kubernetes)、服务网格(Istio)、无服务(Serverless)架构风格实现的其他版本,在业务功能上的表现是完全一致的。如果你不是针对性地带着解决某个具体问题、了解某项具体工具、技术的目的而来,而是有较充裕的时间,希望了解软件架构的全貌与发展的话,笔者推荐以此工程入手来探索现代软件架构,因为单体架构的结构是相对直观的易于理解的,对后面接触的其他架构风格也起良好的铺垫作用。

    以下几种途径,可以运行程序,浏览最终的效果:

    • 通过 Docker 容器方式运行:

      然后在浏览器访问:,系统预置了一个用户(),也可以注册新用户来测试。 默认会使用 HSQLDB 的内存模式作为数据库,并在系统启动时自动初始化好了 Schema,完全开箱即用。但这同时也意味着当程序运行结束时,所有的数据都将不会被保留。 如果希望使用 HSQLDB 的文件模式,或者其他非嵌入式的独立的数据库支持的话,也是很简单的。以常用的 MySQL/MariaDB 为例,程序中也已内置了 MySQL 的表结构初始化脚本,你可以使用环境变量PROFILES来激活 Spring Boot 中针对 MySQL 所提供的配置,命令如下所示:

    • 通过 Git 上的源码,以 Maven 运行:

      然后在浏览器访问:http://localhost:8080单体架构:Spring Boot - 图10,系统预置了一个用户(user:icyfenix,pw:123456),也可以注册新用户来测试。

    • 通过 Git 上的源码,在 IDE 环境中运行:

      • 以 IntelliJ IDEA 为例,Git 克隆本项目后,在 File -> Open 菜单选择本项目所在的目录,或者 pom.xml 文件,以 Maven 方式导入工程。
      • IDEA 将自动识别出这是一个 SpringBoot 工程,并定位启动入口为 BookstoreApplication,待 IDEA 内置的 Maven 自动下载完所有的依赖包后,运行该类即可启动。
      • 可通过 IDEA 的 Maven 面板中 Lifecycle 里面的 package 来对项目进行打包、发布。
      • 在 IDE 环境中修改配置(如数据库等)会更加简单,具体可以参考工程中和application-mysql.yml中的内容。

    Fenix’s Bookstore 单体架构后端尽可能采用标准的技术组件进行构建,不依赖于具体的实现,包括:

    • (JAX-RS 2.1)
      RESTFul 服务方面,采用的实现为 Jersey 2,亦可替换为 Apache CXF、RESTeasy、WebSphere、WebLogic 等。

    • JSR 330:Dependency Injection for Java 1.0单体架构:Spring Boot - 图12
      依赖注入方面,采用的的实现为 SpringBoot 2 中内置的 Spring Framework 5。虽然在多数场合中尽可能地使用了 JSR 330 的标准注解,但仍有少量地方由于 Spring 在对@Named、@Inject 等注解的支持表现上与本身提供的注解差异,使用了 Spring 的私有注解。如替换成其他的 CDI 实现,如 HK2,需要较大的改动。


    • 数据验证方面,采用的实现为 Hibernate Validator 6,可替换为 Apache BVal 等其他验证框架。

    • JSR 315:Java Servlet 3.0单体架构:Spring Boot - 图15
      Web 访问方面,采用的实现为 SpringBoot 2 中默认的 Tomcat 9 Embed,可替换为 Jetty、Undertow 等其他 Web 服务器。

    有以下组件仍然依赖了非标准化的技术实现,包括:


    • 认证/授权方面,在 2017 年才发布的 JSR 375 中仍然没有直接包含 OAuth2 和 JWT 的直接支持,因后续实现微服务架构时对比的需要,单体架构中选择了 Spring Security 5 作为认证服务,Spring Security OAuth 2.3 作为授权服务,Spring Security JWT 作为 JWT 令牌支持,并未采用标准的 JSR 375 实现,如 Soteria。

    • JSR 353/367:Java API for JSON Processing/Binding单体架构:Spring Boot - 图17
      在 JSON 序列化/反序列化方面,由于 Spring Security OAuth 的限制(使用 JSON-B 作为反序列化器时的结果与 Jackson 等有差异),采用了 Spring Security OAuth 默认的 Jackson,并未采用标准的 JSR 353/367 实现,如 Apache Johnzon、Eclipse Yasson 等。

    Fenix’s Bookstore 单体架构后端参考(并未完全遵循)了 DDD 的分层模式和设计原则,整体分为以下四层:

    1. Application:对应 DDD 中的 Application 层,负责定义软件本身对外暴露的能力,即软件本身可以完成哪些任务,并负责对内协调领域对象来解决问题。根据 DDD 的原则,应用层要尽量简单,不包含任何业务规则或者知识,而只为下一层中的领域对象协调任务,分配工作,使它们互相协作,这一点在代码上表现为 Application 层中一般不会存在任何的条件判断语句。在许多项目中,Application 层都会被选为包裹事务(代码进入此层事务开始,退出此层事务提交或者回滚)的载体。
    2. Domain:对应 DDD 中的 Domain 层,负责实现业务逻辑,即表达业务概念,处理业务状态信息以及业务规则这些行为,此层是整个项目的重点。
    3. Infrastructure:对应 DDD 中的 Infrastructure 层,向其他层提供通用的技术能力,譬如持久化能力、远程服务通讯、工具集,等等。
    • 本文档代码部分采用进行许可。遵循许可的前提下,你可以自由地对代码进行修改,再发布,可以将代码用作商业用途。但要求你:
      • 署名:在原有代码和衍生代码中,保留原作者署名及代码来源信息。
      • 保留许可证:在原有代码和衍生代码中,保留 Apache 2.0 协议文件。
    • 本作品文档部分采用知识共享署名 4.0 国际许可协议单体架构:Spring Boot - 图20进行许可。 遵循许可的前提下,你可以自由地共享,包括在任何媒介上以任何形式复制、发行本作品,亦可以自由地演绎、修改、转换或以本作品为基础进行二次创作。但要求你:
      • 署名:应在使用本文档的全部或部分内容时候,注明原作者及来源信息。
      • 非商业性使用:不得用于商业出版或其他任何带有商业性质的行为。如需商业使用,请联系作者。