大家最近有没有注意到?当我们创建一个新的Spring Boot项目时,现在有了Spring Boot 4.0.0预览版的选项,如下图所示:

Spring Boot 4是基于全新的Spring Framework 7.0.0构建的,如下图所示:

那么,问题来了:新版本有哪些重要变化?接下来,让我为大家先睹为快。

1. 优雅的版本控制

新版本引入了对API版本控制的优雅支持,允许开发者通过@RequestMapping注解中的version参数来实现版本控制,如下面的代码所示:

import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/api")
public class VersionedController {

    @RequestMapping(value = "/user", version = "1")
    public String getUserV1() {
        // 版本1实现
        System.out.println("Version 1");
        return "Version 1";
    }

    @RequestMapping(value = "/user", version = "2")
    public String getUserV2() {
        // 版本2实现
        System.out.println("Version 2");
        return "Version 2";
    }
}

程序执行效果:

2. 便捷的Bean注入

新版本引入了新的BeanRegistrar契约,允许更灵活的Bean注册(一次注册多个Bean)。示例代码如下:

import org.springframework.beans.factory.BeanRegistrar;
import org.springframework.beans.factory.BeanRegistry;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;
import org.springframework.core.env.Environment;

@Configuration
@Import(MyBeansRegistrar.class)
public class MyConfiguration {
}

class MyBeansRegistrar implements BeanRegistrar {

    @Override
    public void register(BeanRegistry registry,
                           Environment env) {
        registry.registerBean("user", User.class);
        if (env.matchesProfiles("dev")) {
            registry.registerBean(Order.class, spec -> spec
                    .supplier(context -> new Order("order_001")));
        }
    }
}

class User {
    private String name;
}

class Order {
    public Order(String name) {
        this.name = name;
    }

    private String name;
}

编写单元测试来证明注册的Bean可以正常使用:

3. 空值安全改进

新版本采用JSpecify注解来声明其API的空值安全性,使用@Nullable表示值可以为空,使用@NonNull表示不能为空。这与IntelliJ IDEA配合使用可以提供警告或错误消息,如下面的代码所示:

import org.jspecify.annotations.NonNull;
import org.jspecify.annotations.Nullable;

public class Person {
    private String name;

    public void setName(@NonNull String name) {
        this.name = name;
    }

    @Nullable
    public String getName() {
        return this.name;
    }
}

在IntelliJ IDEA中也可以看到警告消息:

不同IDEA版本的显示格式有所不同;作者使用的是IDEA 2024,因为需要较新的版本才能支持新功能。

4. 轻松创建HTTP代理

新版本提供了@ImportHttpServices注解,使创建HTTP接口代理变得更加容易。示例代码如下(它允许您轻松声明、检测和配置整个HTTP服务组):

@Configuration(proxyBeanMethods = false)
@ImportHttpServices(group = "weather", types = {FreeWeather.class, CommercialWeather.class})
@ImportHttpServices(group = "user", types = {UserServiceInternal.class, UserServiceOfficial.class})
static class HttpServicesConfiguration extends AbstractHttpServiceRegistrar {
    @Bean
    public RestClientHttpServiceGroupConfigurer groupConfigurer() {
        return groups -> groups.filterByName("weather", "user")
                .configureClient((group, builder) -> builder.defaultHeader("User-Agent", "My-Application"));
    }
}

5. 其他七项变化

其他升级功能包括:

  • SPEL表达式升级: SPEL表达式现在支持空值安全和Elvis操作符,如下面的代码所示:

@Value("#{systemProperties['pop3.port'] ?: 25}")

这会注入pop3.port系统属性的值,如果该属性未定义,则注入值25

  • GraalVM原生应用支持: 通过Spring AOT技术,应用程序可以编译为原生镜像,显著减少启动时间。

  • 支持Jackson 3.x: 已放弃对Jackson 2.x的支持,升级到Jackson 3.x。

  • Servlet和WebSocket版本升级: 它使用Servlet 6.1和WebSocket 2.2作为Web应用程序的底层实现。这意味着应用程序应该部署在最新的Servlet容器上,如Tomcat 11+和Jetty 12.1+。

  • HttpHeaders优化: HttpHeaders操作如下:

@RestController
public class MyController {

 @GetMapping("/headers")
 public ResponseEntity<String> handleRequest(HttpHeaders headers) {
 // 旧方式(已弃用)
 // headers.getFirst("X-Custom-Header");

 // 新方式
 String value = headers.firstValue("X-Custom-Header").orElse(null);

 // 遍历所有头部
 headers.forEach((name, values) -> {
 System.out.println(name + ": " + values);
 });

 return ResponseEntity.ok("Processed");
 }
}
  • 功能移除:

  • Spring MVC的XML配置命名空间现已弃用,改为使用Java配置。

  • Spring TestContext Framework中的JUnit 4支持现已弃用。

  • Jackson 2.x支持已弃用,改为使用Jackson 3.x。

  • Spring JCL已停止使用。

  • 最低环境要求提高:

  • Jakarta EE 11 (Tomcat 11+)

  • Kotlin 2.x

  • JSONassert 2.0

  • GraalVM 23

更多升级信息,请参考官方网站:

https://github.com/spring-projects/spring-framework/wiki/Spring-Framework-7.0-Release-Notes

总结

编程是一个终身学习的职业。框架和工具的变化意味着我们获得了更便捷、更用户友好的方法,这对开发者来说是有益的。

所以,让我们一起学习、使用和实验它们吧。