使用 OpenRewrite 简化 Java 和 SpringBoot 迁移
大家好,这里是架构资源栈!点击上方关注,添加“星标”,一起学习大厂前沿架构!

移民的挑战
随着 Spring Boot 2.x 等旧版本即将到期且不再获得支持,迁移到较新版本对于安全性、兼容性和性能改进至关重要。但是,迁移过程面临着一些挑战:
1. 重大变更:主要版本升级通常会引入重大变更。例如,Spring Boot 3.x 需要Java 17并从 迁移javax.*到jakarta.* packages。
**2. 弃用的 API:**许多常用的 API 和模式已被弃用并需要替换。
**3. 手动更新:**传统迁移需要手动更新依赖项、重构代码和修复兼容性问题。
**4. 耗时:**大型代码库可能需要数周或数月才能迁移,从而增加项目成本和风险。
**5. 测试负担:**必须彻底测试每个更改以确保功能保持完好。
那么,我们如何才能简化和加速迁移过程呢?这就是 OpenRewrite 发挥作用的地方。
开放重写
OpenRewrite是一款开源的自动化代码重构工具,可帮助开发人员减少技术负担。它为框架迁移、安全修复和代码样式提供了预构建的重构方案,将工作量从几小时缩短到几分钟。
Gradle 和 Maven 插件可轻松将这些更改应用于存储库。OpenRewrite 社区最初专注于 Java,目前正在积极扩展对更多语言和框架的支持。
主要特点
- **自动重构:**自动更新代码语法、依赖关系和模式
- **基于配方:**使用声明性配方来定义转换规则
- **风格保存:**保留原始代码格式和注释
- **大规模变更:**可以一致地处理整个代码库
- **可扩展:**支持针对特定迁移需求的自定义配方
它是如何工作的?
-
OpenRewrite 修改代表源代码的无损语义树 (LST),并将它们转换回源代码。
-
您可以查看更改并根据需要提交。
-
修改是使用访问者进行的,访问者被分组到菜谱中。
-
配方可确保变化最小,并保持原始格式。
实践
在本文中,我将演示如何使用 OpenRewrite 将使用 Java 8、Spring Boot 2.x 和 JUnit 4 构建的简单 CRUD Spring Boot 应用程序迁移到 Java 21、Spring Boot 3.3 和 JUnit 5。
代码库
1)pom.xml
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.7.14</version><relativePath/></parent><groupId>com.example</groupId><artifactId>demo</artifactId><version>0.0.1-SNAPSHOT</version><name>demo</name><description>Demo project for Spring Boot Migration</description><properties><java.version>1.8</java.version></properties><dependencies>// dependencies: starter web, data-jpa, etc</dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build>
</project>
Enter fullscreen mode Exit fullscreen mode
2)UserController.java
@RestController
@RequestMapping("/api/users")
public class UserController {@Autowiredprivate UserRepository userRepository;@Autowiredprivate UserService userService;@RequestMapping(method = RequestMethod.GET)public List<User> getAllUsers() {return userRepository.findAll();}@RequestMapping(method = RequestMethod.POST)public ResponseEntity<?> createUser(@Valid @RequestBody User user) {User savedUser = userRepository.save(user);return ResponseEntity.ok().build();}@RequestMapping(value = "/{id}", method = RequestMethod.GET)public ResponseEntity<User> getUserById(@PathVariable("id") Long id) {User user = userRepository.findById(id).orElse(null);return user != null ? ResponseEntity.ok(user) : ResponseEntity.notFound().build();}@RequestMapping(value = "/username")public ResponseEntity<User> getUserByUsername(@RequestParam String username) {User user = userService.findByUsername(username);return user != null ? ResponseEntity.ok(user) : ResponseEntity.notFound().build();}}
Enter fullscreen mode Exit fullscreen mode
3)UserService.java
@Service
public class UserService {@Autowiredprivate UserRepository userRepository;public User findByUsername(String username) {return userRepository.findByUsernameNative(username);}
}
Enter fullscreen mode Exit fullscreen mode
4)UserRepository.java
@Repository
public interface UserRepository extends JpaRepository<User, Long> {@Query(value = "SELECT * FROM users WHERE username = ?1", nativeQuery = true)User findByUsernameNative(String username);}
Enter fullscreen mode Exit fullscreen mode
5)用户.java
import javax.persistence.*;
import javax.validation.constraints.NotNull;@Entity
@Table(name = "users")
public class User {@Id@GeneratedValue(strategy = GenerationType.AUTO)private Long id;@NotNull@Column(nullable = false)private String username;@Columnprivate String email;// setter, getter
}
Enter fullscreen mode Exit fullscreen mode
6)UserControllerTest.java
@RunWith(SpringRunner.class)
@SpringBootTest
@AutoConfigureMockMvc
public class UserControllerTest {@Autowiredprivate MockMvc mockMvc;@Autowiredprivate UserRepository userRepository;@Beforepublic void setup() {userRepository.deleteAll();}@Testpublic void testCreateUser() throws Exception {String userJson = "{"username":"testuser","email":"test@example.com"}";mockMvc.perform(MockMvcRequestBuilders.post("/api/users").contentType(MediaType.APPLICATION_JSON).content(userJson)).andExpect(MockMvcResultMatchers.status().isOk());}@Testpublic void testGetUser() throws Exception {User user = new User();user.setUsername("testuser");user.setEmail("test@example.com");userRepository.save(user);mockMvc.perform(MockMvcRequestBuilders.get("/api/users/" + user.getId())).andExpect(MockMvcResultMatchers.status().isOk()).andExpect(MockMvcResultMatchers.jsonPath("$.username").value("testuser"));}
}
Enter fullscreen mode Exit fullscreen mode
手动迁移
在使用 OpenRewrite 之前,让我们看看如果手动迁移到 Java 21 和 Spring Boot 3.3 会发生什么。
首先,更新pom.xml
<parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>3.3.10</version><relativePath/>
</parent><properties><java.version>21</java.version>
</properties>
Enter fullscreen mode Exit fullscreen mode
接下来,使用 Maven 清理并构建项目:
mvn clean install
有几个编译错误

好的,让我们恢复更改并使用 OpenRewrite 进行迁移。
使用 OpenRewrite 进行迁移
添加 OpenRewrite 插件
在 中pom.xml,添加 OpenRewrite Maven 插件。如果您使用的是 Gradle,则可以添加 Gradle 插件。
<build><plugins><plugin><groupId>org.openrewrite.maven</groupId><artifactId>rewrite-maven-plugin</artifactId><version>6.3.2</version></plugin></plugins>
</build>
Enter fullscreen mode Exit fullscreen mode
选择迁移方案
要发现所有可用的配方,您可以检查配方目录,其中列出了所有可用的配方,包括:Java、Spring Boot、Hibernate、Quarkus、Scala、.NET、Jenkins 等。
在这个展示中,我将使用三个配方:Java 8 到 21、Spring 2 到 3 和 JUnit 4 到 5。
让我们将食谱添加到 pom.xml 中;每个食谱都有自己的依赖项。
<plugin><groupId>org.openrewrite.maven</groupId><artifactId>rewrite-maven-plugin</artifactId><version>6.3.2</version><configuration><exportDatatables>true</exportDatatables><activeRecipes><recipe>org.openrewrite.java.migrate.UpgradeToJava21</recipe><recipe>org.openrewrite.java.spring.boot2.SpringBoot2JUnit4to5Migration</recipe><recipe>org.openrewrite.java.spring.boot3.UpgradeSpringBoot_3_3</recipe></activeRecipes></configuration><dependencies><dependency><groupId>org.openrewrite.recipe</groupId><artifactId>rewrite-migrate-java</artifactId><version>3.4.0</version></dependency><dependency><groupId>org.openrewrite.recipe</groupId><artifactId>rewrite-spring</artifactId><version>6.3.0</version></dependency></dependencies>
</plugin>
Enter fullscreen mode Exit fullscreen mode
现在,运行 Maven 来安装 OpenRewrite 插件及其配方:
mvn clean install
预览迁移
OpenRewrite 提供了一种dryRun模式,允许开发人员在实际应用更改之前通过以下方式预览更改:
mvn rewrite:dryRun
您可以看到将用于实际迁移的更改。

应用迁移
现在,让我们进行迁移
mvn rewrite:run
使用您的 IDE 或差异检查工具来检查更改。
1.更新pom.xml:自动更新SpringBoot版本到3.3.10,Java版本到21,删除JUnit4。

2. 更新控制器:
从 迁移javax.*到jakarta.* packages,改为使用@GetMapping,@PostMapping用于专用注释。

3. 更新单元测试:
替换@Before为@BeforeEach,更新包名称

限制
在这个展示中,OpenRewrite 成功迁移到 Java 21 和 Spring Boot 3.3,但它仍然存在一些限制。
由于 OpenRewrite 依赖于预定义的配方,因此它支持许多常见框架,但并非全部。
例如,如果您需要将第三方库(如 Ehcache2)迁移到 Ehcache3(Spring 3.0 中不再支持),OpenRewrite 不提供内置配方。在这种情况下,您必须编写自定义配方或手动执行迁移。
如果您创建了自定义配方,请考虑将其贡献给 OpenRewrite 社区,以帮助其他进行类似迁移的人。
概括
OpenRewrite 通过以下方式显著简化了 Java 和 Spring Boot 迁移过程:
- 自动重复代码更改
- 减少迁移时间和精力
- 最大限度地减少人为错误
- 标准化迁移方法
虽然 OpenRewrite并不能消除测试和验证的需要,但它减少了迁移所需的手动工作量。这使开发人员可以专注于更复杂的迁移方面和业务逻辑更新。
参考
- https://docs.openrewrite.org/
- https://github.com/openrewrite
原文地址:https://mp.weixin.qq.com/s/c7HMXbTJdxYWj53tKdY3hA
相关文章:
使用 OpenRewrite 简化 Java 和 SpringBoot 迁移
大家好,这里是架构资源栈!点击上方关注,添加“星标”,一起学习大厂前沿架构! 移民的挑战 随着 Spring Boot 2.x 等旧版本即将到期且不再获得支持,迁移到较新版本对于安全性、兼容性和性能改进至关重要。但…...
2025中国移动云智算大会回顾:云智变革,AI+跃迁
4月10日,2025中国移动云智算大会在苏州举办。会上,中国移动开启“由云向智”新范式,以“智”为核心开辟算网新生态,彰显其在AI新时代的战略远见与技术引领力。 “云智算”将如何通过算网基础设施与人工智能核心技术的深度融合&am…...
ubuntu 2204 安装 vcs 2023
系统 : Ubuntu 22.04.1 LTS vcs 软件 : 有已经安装好的软件(位于redhat8.10),没找到安装包 . 安装好的目录文件 占用 94 G注意 : 该虚拟机(包括安装好的软件)有114G,其中安装好的目录文件占用94GB // 即 我要把 这里 已经安装好的软件(包括scl/vcs/verdi 和其他软件) 在 …...
C#实例化类型详解:从概念到实战
在C#编程中,实例化类型是构建程序的核心操作。本文将通过通俗易懂的案例,深入解析类型实例化的原理与实践技巧,帮助开发者快速掌握这一基础但至关重要的概念。 实例化类型是什么? 定义 通过类型模板创建具体对象的过程称为实例…...
Redis的Key的过期策略
我们都知道Redis的键值对是可以设置过期时间的,那么就会涉及到一个问题,Redis到底是如何做到响应快的同时有能快速地释放掉过期的键值对的呢?不卖关子了,直接说答案,那就是Redis两个策略:定期删除和惰性删除…...
python爬虫降低IP封禁,python爬虫除了使用代理IP和降低请求频率,还有哪些方法可以应对IP封禁?
文章目录 前言1. 利用 CDN 节点2. 模拟真实用户行为3. 使用 IP 池轮换策略4. 处理 Cookie 和会话信息5. 分布式爬虫 前言 除了使用代理 IP 和降低请求频率,以下这些方法也能应对 IP 封禁: Python 3.13.2安装教程(附安装包)Python…...
Unity URP Moblie AR示例工程,真机打包出来,没阴影
效果: unity ar示例演示 现象: 真机打包测试私活没有阴影 Unity版本:2022.3.4f1c1 分析原因: Prefab :ARFeatheredPlane中也有材质,一个用于环境遮挡,一个用于阴影接受。 按理说有啊。 urp …...
单片机 | 基于51单片机的自动循迹小车设计
以下是一个基于51单片机的自动循迹小车设计详解,包含原理、公式和完整代码: 一、系统原理 核心模块: 传感器:红外对管(TCRT5000)x4主控芯片:STC89C52RC(51单片机)电机驱动:L298N驱动模块电源:7.4V锂电池(电机) + 5V稳压(单片机)工作原理: 红外对管发射红外线,…...
【AI】——结合Ollama、Open WebUI和Docker本地部署可视化AI大语言模型
🎼个人主页:【Y小夜】 😎作者简介:一位双非学校的大三学生,编程爱好者, 专注于基础和实战分享,欢迎私信咨询! 🎆入门专栏:🎇【MySQL࿰…...
Hook插件
hook插件 1.概念 在JavaScript中,hook是一种能够拦截和修改函数或方法行为的技术。通过使用hook,开发者可以在现有的函数执行前、执行后或者替换函数的实现逻辑。hook目的是找到函数入口以及一些参数变化,便于分析js逻辑。 2.hook的作用&a…...
双 Token 与 单 Token 优缺点
双Token与单Token认证机制对比 在Web应用开发中,身份认证和授权是保障系统安全的核心环节。随着技术演进,基于Token的认证机制逐渐取代传统Session方案,而双Token与单Token架构的选型争议也日益成为开发者关注的焦点。本文将从技术原理、优缺…...
Android 不插SIM卡,手机不能拨打紧急电话;2g+gsm配置才支持112紧急拨号
[DESCRIPTION] 不插SIM卡,手机不能拨打紧急电话 Root Cause 手机没有写入合法的IMEI;或者当地的某个运营商不支持紧急电话,而手机正好选上了这个运营商;或者当地的某个运营商不支持无SIM卡的紧急电话,而手机正好选上了这个运营商 [SOLUTION] …...
java 多线程之Worker Thread模式(Thread Pool模式)
Worker Thread模式 Worker的意思是工作的人,在Worker Thread模式中,工人线程Worker thread会逐个取回工作并进行处理,当所有工作全部完成后,工人线程会等待新的工作到来。 Worker Thread模式也被成为Background Threadÿ…...
在 MoonBit 中引入 Elm 架构:用简单原则打造健壮的 Web 应用
Elm 是一种纯函数式编程语言,专为构建前端 Web 应用程序而设计。它编译为 JavaScript,强调简洁性、性能和健壮性。 纯函数式的含义是函数没有副作用,这使得代码更易于理解和调试。通过强大的静态类型检查,Elm 确保应用程序不会抛…...
CMD命令行笔记
CMD命令行笔记,涵盖常用命令及实用技巧,适合快速查阅: 一、基础操作 打开CMD Win R → 输入 cmd → 回车管理员模式:右键开始菜单 → 选择“命令提示符(管理员)” 常用命令 help:查看所有命令…...
Python自动化办公
第五篇:Python自动化办公:10行代码搞定重复性工作 适合读者:职场人士、数据分析师 | 阅读时长:12分钟 引言 每天重复处理Excel、PDF或邮件?Python可以帮你自动化这些枯燥任务,节省90%的时间。本文通过实际…...
PDF 转换为 Word、HTML、LaTeX 和 Markdown 格式
PDF 转换为 Word、HTML、LaTeX 和 Markdown 格式 1. Doc2XReferences https://doc2x.com/ 1. Doc2X References [1] Yongqiang Cheng, https://yongqiang.blog.csdn.net/ [2] GPT 学术优化 (GPT Academic), https://github.com/binary-husky/gpt_academic [3] 学术版 GPT 网页…...
C#中async await异步关键字用法和异步的底层原理
目录 C#异步编程一、异步编程基础二、异步方法的工作原理三、代码示例四、编译后的底层实现五、总结 C#异步编程 一、异步编程基础 异步编程是啥玩意儿 就是让程序在干等着某些耗时操作(比如等网络响应、读写文件啥的)的时候,能把线程腾出来…...
shardingsphere-jdbc集成Seata分布式事务
1、导入相关依赖 <!-- shardingsphere-jdbc --><dependency><groupId>org.apache.shardingsphere</groupId><artifactId>shardingsphere-jdbc</artifactId><version>5.5.1</version></dependency><!-- shardingspher…...
华为OD机试真题——统计匹配的二元组个数(2025A卷:100分)Java/python/JavaScript/C++/C语言/GO六种最佳实现
2025 A卷 100分 题型 本文涵盖详细的问题分析、解题思路、代码实现、代码详解、测试用例以及综合分析; 并提供Java、python、JavaScript、C、C语言、GO六种语言的最佳实现方式! 2025华为OD真题目录全流程解析/备考攻略/经验分享 华为OD机试真题《统计匹配…...
CSS 表格样式学习笔记
CSS 提供了强大的工具来美化和定制 HTML 表格的外观。通过合理使用 CSS 属性,可以使表格更加美观、易读且功能强大。以下是对 CSS 表格样式的详细学习笔记。 一、表格边框 1. 单独边框 默认情况下,表格的 <table>、<th> 和 <td> 元…...
MySQL表的增删改查进阶版
Mysql 1、数据库的约束1.1约束类型1.2 NULL约束1.3 UNIQUE:唯一约束1.4 DEFAULT:默认值约束1.5 PRIMARY KEY:主键约束(重点)1.6 FOREIGN KEY:外键约束(重点) 2.表的设计2.1一对一2.2…...
记录 | Pycharm中如何调用Anaconda的虚拟环境
目录 前言一、步骤Step1 查看anaconda 环境名Step2 Python项目编译器更改 更新时间 前言 参考文章: 参考视频:如何在pycharm中使用Anaconda创建的python环境 自己的感想 这里使用的Pycharm 2024专业版的。我所使用的Pycharm专业版位置:【仅用…...
2025年K8s最新高频面试题
目录 Kubernetes的核心组件有哪些,各自作用是什么? Pod和Deployment的区别? Service有哪些类型,分别适用于什么场景? ConfigMap和Secret有什么区别? StatefulSet 和 Deployment 的主要区别是什么? 什么是 Ingress,有哪些常用实现方式? 如何限制 Kubernetes 中 Pod …...
【Android】LiveData深度解析
一,概述 1,LiveData是状态订阅组件,是粘性的,而非事件订阅组件(可以没有事件,但不能没有状态)。所谓的状态,即UI状态,同一时刻只存在一种,且是最新状态,过期的状态应该被遗弃。事件,则是生产者创建的事件,需一一消费,不能被遗弃。 2,Android页面承载组件Activ…...
数据结构专题 - 线性表
线性表是数据结构中最基础、最常用的数据结构之一,它在实际应用中非常广泛。无论是操作系统中的内存管理,还是数据库中的索引结构,线性表都扮演着重要角色。 一、线性表的概念与抽象数据类型 1.1 线性表的逻辑结构 线性表是由n(…...
上门送水小程序区域代理模块框架设计
一、逻辑分析 代理申请流程: 潜在代理商通过小程序提交代理申请,需要填写个人或企业基本信息、联系方式、期望代理区域等。系统收到申请后,进行初步审核,检查信息的完整性和合规性。运营人员进行人工审核,根据公司政策…...
asp-for等常用的HTML辅助标记?
在ASP.NET Core Razor Pages 和 MVC 中,除了asp-for之外,还有许多常用的 HTML 辅助标记,下面为你详细介绍: 表单与路由相关 asp-action 和 asp-controller 用途:这两个标记用于生成表单或链接的 URL,指定…...
qt pyqt5的开发, 修改psd图像
这是引子, 需要将这个 photoshop-python-api 进行使用 https://juejin.cn/post/7445112318693621797#heading-4 这个是ps-python-api的官网, 在里面找api文档 https://pypi.org/project/photoshop-python-api/ 源码.gitee.url https://gitee.com/lbnb/psd_work.git 一. 安装必要…...
Spring 中的循环依赖问题:解决方案与三级缓存机制
目录 Spring 中的循环依赖问题:解决方案与三级缓存机制什么是循环依赖?循环依赖的定义循环依赖的举例 Spring 中的循环依赖类型1. 构造器注入引发的循环依赖2. Setter 注入引发的循环依赖3. 字段注入(Autowired)引发的循环依赖 Sp…...
