DevNotes-SpringBoot生产中16条最佳实践-《Java笔记》

admin 2025-10-19 04:54:26 编程 来源:ZONE.CI 全球网 0 阅读模式

Java SpringBootSpring Boot是最流行的用于开发微服务的Java框架。这里分享在开发中使用Spring Boot所采用的最佳实践。重点介绍Spring Boot特有的实践(大多数时候,也适用于Spring项目)。以下依次列出了最佳实践,排名不分先后。

1、使用自定义BOM来维护第三方依赖

Spring Boot项目本身使用和集成了大量的开源项目,它维护了这些第三方依赖。但是也有一部分在实际项目使用中并没有包括进来,这就需要在项目中自己维护版本。如果在一个大型的项目中,包括了很多未开发模块,那么维护起来就非常的繁琐。怎么办呢?事实上,Spring IO Platform就是做的这个事情,它本身就是Spring Boot的子项目,同时维护了其他第三方开源库。可以借鉴Spring IO Platform来编写自己的基础项目platform-bom,所有的业务模块项目应该以BOM的方式引入。这样在升级第三方依赖时,就只需要升级这一个依赖的版本而已。

  1. <dependencyManagement>
  2. <dependencies>
  3. <dependency>
  4. <groupId>io.spring.platform</groupId>
  5. <artifactId>platform-bom</artifactId>
  6. <version>Cairo-SR3</version>
  7. <type>pom</type>
  8. <scope>import</scope>
  9. </dependency>
  10. </dependencies>
  11. </dependencyManagement>

2、使用自动配置

Spring Boot的一个主要特性是使用自动配置。这是Spring Boot的一部分,它可以简化代码并使之工作。当在类路径上检测到特定的jar文件时,自动配置就会被激活。使用它的最简单方法是依赖Spring Boot Starters。因此,如果想与Redis进行集成,可以首先包括:

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-data-redis</artifactId>
  4. </dependency>

如果想与MongoDB进行集成,需要这样:

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-data-mongodb</artifactId>
  4. </dependency>

借助于这些starters,这些繁琐的配置就可以很好地集成起来并协同工作,而且它们都是经过测试和验证的。这非常有助于避免可怕的Jar地狱。https://dzone.com/articles/what-is-jar-hell通过使用以下注解属性,可以从自动配置中排除某些配置类:

  1. @EnableAutoConfigurationexclude = {ClassNotToAutoconfigure.class})

但只有在绝对必要时才应该这样做。有关自动配置的官方文档可在此处找到:https://docs.spring.io/spring-boot/docs/current/reference/html/using-boot-auto-configuration.html

3、使用Spring Initializr来开始一个新的Spring Boot项目

Spring Initializr 提供了一个超级简单的方法来创建一个新的Spring Boot项目,并根据需要来加载可能使用到的依赖。https://start.spring.io/使用Initializr创建应用程序可确保获得经过测试和验证的依赖项,这些依赖项适用于Spring自动配置。甚至可能会发现一些新的集成。

4、考虑为常见的组织问题创建自己的自动配置

这一条也来自Josh Long(Spring Advocate,@starbuxman)——这个实践是针对高级用户的。如果在一个严重依赖Spring Boot的公司或团队中工作,并且有共同的问题需要解决,那么可以创建自己的自动配置。这项任务涉及较多工作,因此需要考虑何时获益是值得投入的。与多个略有不同的定制配置相比,维护单个自动配置更容易。如果将这个提供Spring Boot配置以开源库的形式发布出去,那么将极大地简化数千个用户的配置工作。

5、正确设计代码目录结构

避免使用默认包。确保所有内容(包括入口点)都位于一个名称很好的包中,这样就可以避免与装配和组件扫描相关的意外情况;将Application.java(应用的入口类)保留在顶级源代码目录中;建议将控制器和服务放在以功能为导向的模块中,但这是可选的。一些非常好的开发人员建议将所有控制器放在一起。不论怎样,坚持一种风格!

6、保持@Controller的简洁和专注

Controller应该非常简单。可以在此处阅读有关GRASP中有关控制器模式部分的说明。希望控制器作为协调和委派的角色,而不是执行实际的业务逻辑。以下是主要做法:https://en.wikipedia.org/wiki/GRASP(object-orienteddesign)#Controller#Controller)1、控制器应该是无状态的!默认情况下,控制器是单例,并且任何状态都可能导致大量问题;2、控制器不应该执行业务逻辑,而是依赖委托;3、控制器应该处理应用程序的HTTP层,这不应该传递给服务;4、控制器应该围绕用例/业务能力来设计。要深入这个内容,需要进一步地了解设计REST API的最佳实践。无论是否想要使用Spring Boot,都是值得学习的。

7、围绕业务功能构建@Service

Service是Spring Boot的另一个核心概念。最好围绕业务功能/领域/用例(无论怎么称呼都行)来构建服务。在应用中设计名称类似 AccountService, UserService, PaymentService这样的服务,比起像 DatabaseService、 ValidationService、 CalculationService这样的会更合适一些。可以决定使用Controller和Service之间的一对一映射,那将是理想的情况。但这并不意味着,Service之间不能互相调用!

8、使数据库独立于核心业务逻辑之外

之前还不确定如何在Spring Boot中最好地处理数据库交互。在阅读了罗伯特·C·马丁的“Clear Architecture”之后,就清晰多了。希望数据库逻辑于服务分离出来。理想情况下,不希望服务知道它正在与哪个数据库通信,这需要一些抽象来封装对象的持久性。罗伯特C.马丁强烈地说明,数据库是一个“细节”,这意味着不将应用程序与特定数据库耦合。过去很少有人会切换数据库,使用Spring Boot和现代微服务开发会让事情变得更快。

9、保持业务逻辑不受Spring Boot代码的影响

考虑到“Clear Architecture”的教训,还应该保护业务逻辑。将各种Spring Boot代码混合在一起是非常诱人的……不要这样做。如果能抵制诱惑,将保持业务逻辑可重用。部分服务通常成为库。如果不从代码中删除大量Spring注解,则更容易创建。

10、推荐使用构造函数注入

保持业务逻辑免受Spring Boot代码侵入的一种方法是使用构造函数注入。不仅是因为 @Autowired注解在构造函数上是可选的,而且还可以在没有Spring的情况下轻松实例化bean。

11、熟悉并发模型

https://www.e4developer.com/2018/03/30/introduction-to-concurrency-in-spring-boot/在Spring Boot中,Controller和Service是默认是单例。如果不小心,这会引入可能的并发问题。通常也在处理有限的线程池。

12、加强配置管理的外部化

这一点超出了Spring Boot,虽然这是人们开始创建多个类似服务时常见的问题……可以手动处理Spring应用程序的配置。如果正在处理多个Spring Boot应用程序,则需要使配置管理能力更加强大。推荐两种主要方法:1、使用配置服务器,例如Spring Cloud Config;2、将所有配置存储在环境变量中(可以基于git仓库进行配置)。这些选项中的任何一个(第二个选项多一些)都要求在DevOps更少工作量,但这在微服务领域是很常见的。

13、提供全局异常处理

真的需要一种处理异常的一致方法。Spring Boot提供了两种主要方法:1、应该使用HandlerExceptionResolver定义全局异常处理策略;2、也可以在控制器上添加@ExceptionHandler注解,这在某些特定场景下使用可能会很有用。这与Spring中的几乎相同,并且Baeldung有一篇关于REST与Spring的错误处理的详细文章,非常值得一读。https://www.baeldung.com/exception-handling-for-rest-with-spring

14、使用日志框架

应该使用Logger进行日志记录,而不是使用System.out.println()手动执行。这很容易在Spring Boot中完成,几乎没有配置。只需获取该类的记录器实例:

  1. Logger logger = LoggerFactory.getLogger(MyClass.class);

这很重要,因为它可以根据需要设置不同的日志记录级别。

15、测试代码

这不是Spring Boot特有的,但它需要提醒——测试代码!如果没有编写测试,那么将从一开始就编写遗留代码。如果有其他人使用代码库,那边改变任何东西将会变得危险。当有多个服务相互依赖时,这甚至可能更具风险。由于存在Spring Boot最佳实践,因此应该考虑将Spring Cloud Contract用于消费者驱动契约,它将与其他服务的集成更容易使用。

16、使用测试切片让测试更容易,并且更专注

使用Spring Boot测试代码可能很棘手——需要初始化数据层,连接大量服务,模拟事物……实际上并不是那么难!答案是使用测试切片。使用测试切片,可以根据需要仅连接部分应用程序。这可以为节省大量时间,并确保测试不会与未使用的内容相关联。来自spring.io的一篇名为Custom test slice with Spring test 1.4的博客文章解释了这种技术。https://spring.io/blog/2016/08/30/custom-test-slice-with-spring-boot-1-4

以太坊cppgolang区别 编程

以太坊cppgolang区别

以太坊是一种去中心化的开源平台,它采用智能合约技术,旨在构建和运行不受干扰的分布式应用程序。作为目前最受欢迎的区块链平台之一,以太坊提供了多种编程语言的支持,其
progolang 编程

progolang

Go语言(Golang)是由Google开发的一门静态类型编程语言。作为一名专业的Golang开发者,我深知这门语言的优势和特点。在本文中,我将介绍Golang
golangn个发送者 编程

golangn个发送者

Golang是一种开源的编程语言,由Google团队开发,旨在提高程序的并发性和简化软件开发过程。在Go语言中,有时需要向多个接收者发送信息。本文将介绍如何在G
golang技能图谱 编程

golang技能图谱

从互联网行业的快速发展到人工智能技术的日益成熟,各种编程语言也应运而生。而在这众多的编程语言中,Golang(即Go)作为一门强大且高效的开发语言备受关注。Go
评论:0   参与:  0