Spring Boot 是一种广泛使用且非常流行的企业级高性能框架。以下是一些最佳实践和一些技巧,我们可以使用它们来改进 Spring Boot 应用程序并使其更加高效。这篇文章会有点长,完整读完文章需要一些时间。
1.正确的包目录风格
- 正确的包目录将有助于轻松理解代码和应用程序的流程。
- 我们可以使用有意义的包目录来构建我们的应用程序。
- 我们可以将所有控制器包含在单独的包中,将服务包含在单独的包中,将 util 类包含在单独的包中等等。这种风格在小型微服务中非常方便。
- 如果我们正在处理庞大的代码库,则可以使用基于功能模块的方法。我们可以根据我们的要求来决定。
基于类型
基于功能模块
2.使用设计模式
没什么好说的,设计模式已经是现代编程中编写可维护、可扩展代码的最佳实践。
推荐一个开源免费的 Spring Boot 实战项目:
GitHub - javastacks/spring-boot-best-practice: Spring Boot 最佳实践,包括自动配置、核心原理、源码分析、国际化支持、调试、日志集成、热部署等。
3.使用 Spring Boot starter
这是 Spring Boot 的一个很酷的功能。
我们可以非常轻松地使用启动器依赖项,而无需一一添加单个依赖项。这些入门依赖项已与所需的依赖项捆绑在一起。
例如,如果我们添加 spring-boot-starter-web 依赖项,默认情况下它会与 jackson、spring-core、spring-mvc 和 spring-boot-starter-tomcat 依赖项捆绑在一起。
所以我们不需要关心单独添加依赖项。
它还可以帮助我们避免版本不匹配。
4.使用生产版本的依赖项
始终建议使用最新的稳定 GA 版本。
有时它可能会因 Java 版本、服务器版本、应用程序类型等而有所不同。
不要使用同一包的不同版本,如果存在多个依赖项,请始终使用<properties>
指定版本。
5.使用 Lombok
作为一名 Java 开发人员,我们可能听说过 Lombok 项目。
Lombok 是一个 Java 库,可用于减少代码并允许我们使用其注释编写干净的代码。
例如,我们可能在某些类(如实体、请求/响应对象、dtos 等)中使用大量的 getter 和 setter 行。
但如果你使用 Lombok,它只是一行,你可以根据你的要求使用@Data、@Getter 或@Setter。
我们也可以使用 Lombok 记录器注释。推荐@Slf4j。
检查此文件以供参考。
6.将构造函数注入与 Lombok 一起使用
当我们谈论依赖注入时,有两种类型。
一种是“构造函数注入”,另一种是“setter 注入”。除此之外,我们还可以使用非常流行的@Autowired 注释来使用“字段注入”。
但我们强烈建议使用构造函数注入而不是其他类型。因为它允许应用程序在初始化时初始化所有必需的依赖项。
这对于单元测试非常有用。
重要的是,我们可以使用 Lombok 的 @RequiredArgsConstructor 注释来使用构造函数注入。
检查此示例控制器以供参考。
7.使用 slf4j 日志
- 日志记录非常重要。
- 如果我们的应用程序在生产过程中出现问题,日志记录是找出根本原因的唯一方法。
- 因此,在添加记录器、日志消息类型、记录器级别和记录器消息之前应该仔细考虑。
- 不要使用 System.out.print()
- 建议将 Slf4j 与 Spring Boot 中默认的日志框架 logback 一起使用。
- 始终使用 slf4j 的 {} 占位符语法,避免在记录器消息中使用字符串插值。因为字符串插值会消耗更多的内存。
- 我们可以使用 Lombok @Slf4j 注释非常轻松地创建日志记录器。
- 如果我们处于微服务环境中,则可以使用 ELK 技术栈。
8.控制器仅用于路由
- 控制器专用于路由。
- 它是无状态且单身的。
- DispatcherServlet 将检查控制器上的 @RequestMapping
- 控制器是请求的最终目标,请求将交给服务层并由服务层处理。
- 业务逻辑不应位于控制器中。
9.使用Service来实现业务逻辑
- 完整的业务逻辑包含验证、缓存等。
- 与持久层通信并接收结果。
- Service也是单例的。
10.避免空指针异常
- 为了避免 NullPointerException,我们可以使用 java.util 包中的 Optional。
- 我们还可以使用空安全库。例如:Apache Commons StringUtils
- 对已知对象调用 equals() 和 equalsIgnoreCase() 方法。
- 使用 valueOf() 而不是 toString()
- 使用基于 IDE 的 @NotNull 和 @Nullable 注释。
11.使用集合框架的最佳实践
- 对我们的数据集使用适当的集合。
- 将 forEach 与 Java 8 功能结合使用,并避免使用旧版 for 循环。
- 使用接口类型而不是实现。
- 使用 isEmpty() 而不是 size() 以获得更好的可读性。
- 不返回空值,可以返回空集合。
- 如果我们使用对象作为要存储在基于哈希的集合中的数据,则应重写 equals() 和 hashCode() 方法。请查看这篇文章“HashMap 内部是如何工作的”。
12.使用分页
这将提高应用程序的性能。
如果我们使用 Spring Data JPA,则 PagingAndSortingRepository 使分页的使用变得非常容易且几乎不费吹灰之力。
13.使用缓存
在谈论应用程序性能时,缓存是另一个重要因素。
默认情况下,Spring Boot 通过 ConcurrentHashMap 提供缓存,我们可以通过 @EnableCaching 注解来实现这一点。如果我们对默认缓存不满意,可以使用 Redis、Hazelcast 或任何其他分布式缓存实现。
Redis 和 Hazelcast 是内存缓存方法。我们还可以使用数据库缓存实现。
14.使用自定义异常处理程序和全局异常处理
- 这在使用大型企业级应用程序时非常重要。
- 除了一般异常之外,我们可能还会有一些场景来识别某些特定的错误情况。
- 异常顾问可以使用@ControllerAdvice 创建,我们可以创建具有有意义细节的单独异常。
- 它将使得将来识别和调试错误变得更加容易。
15.使用自定义响应对象
- 自定义响应对象可用于返回包含某些特定数据的对象,并满足 HTTP 状态代码、API 代码、消息等要求。
- 我们可以使用构建器设计模式来创建具有自定义属性的自定义响应对象。
16.删除不必要的代码、变量、方法和类。
- 未使用的变量声明将占用一些内存。
- 删除未使用的方法、类等,因为它会影响应用程序的性能。
- 尽量避免嵌套循环。我们可以使用map代替。
17.使用注释
- 注释是一个很好的做法。
- 不要对一切代码发表注释。相反,我们可以使用类、函数、方法、变量等有意义的单词编写描述性代码。
- 删除注释代码、误导性注释和故事型注释。
- 我们可以使用注释进行警告,并解释一些乍一看难以理解的内容。
18.对类、方法、函数、变量和其他属性使用有意义的词语。
- 这看起来很简单,但影响却是巨大的。
- 始终使用正确的有意义且可搜索的命名约定以及正确的大小写。
- 通常,我们在声明类、变量和常量时使用名词或短语。例如:字符串 firstName,const isValid
- 我们可以使用带有形容词的动词和短语来表示函数和方法。例如:readFile()、sendData()
- 避免使用缩写变量名和意图揭示的名称。例如:int i;字符串 getExUsr;
- 如果我们有意义地使用此功能,则可以减少声明注释行。由于它具有有意义的名称,新开发人员可以通过阅读代码轻松理解。
19.使用正确的大小写进行声明
有许多不同的大小写,如大写、小写、驼峰命名、帕斯卡命名、蛇命名、大蛇式命名、短横线命名等。
但我们需要确定哪个案例专用于哪个变量。
通常,我会遵循,
- 类 — 帕斯卡命名
- 方法和变量 — 驼峰命名
- 常量 — 大蛇式命名
- 数据库相关字段 — 短横线命名
这只是一个例子,它可能与我们在公司遵循的标准不同。
20.简单点
- 始终尝试编写简单、可读的代码。
- 同样简单的逻辑可以用不同的方式实现,但是如果不可读或不理解就很难理解。
- 有时复杂的逻辑会消耗更多的内存。
- 编写代码时尝试使用 KISS、DRY 和 SOLID 原则。我将在以后的文章中解释这一点。
21.使用通用的代码格式样式
- 格式样式因开发人员而异。编码风格的改变也被认为是一种改变,并且会使代码合并变得非常困难。
- 为了避免这种情况,团队可以采用通用的编码格式。
22.使用 SonarLint 插件
这对于识别小错误和最佳实践非常有用,以避免不必要的错误和代码质量问题。
我们可以将插件安装到我们最喜欢的 IDE 中。
最后
至此本文讲解内容到此完毕感谢阅读,希望本文能对你有所帮助。