Spring-WebFlux

Spring5新特性与WebFlux介绍

Spring5新特性

1. Spring Framework 5 中文文档:Spring Framework 5 中文文档简介(Spring 5 Reference)

2. Spring5整体代码基于Java8

  • 对java8提高直接的代码支撑
  • 运行时兼容JDK9
  • 许多不建议使用的类和方法在代码库中删除
  • Spring 5.0框架自带了通用的日志封装

3. Spring5配置log4j2

  • Spring5 已经移除Log4jConfigListener
  • 创建 log4j2.xml 配置文件,xml名字不可改变,配置内容固定
<?xml version="1.0" encoding="UTF-8"?>

<!--
	日志级别以及优先级排序: 
		OFF > FATAL > ERROR > WARN > INFO > DEBUG > TRACE > ALL
-->
<!--
	status用于设置log4j2自身内部的信息输出
	可以不设置,
	当设置成trace时,可以看到log4j2内部各种详细输出
-->
<configuration status="INFO">
    
	<!--先定义所有的appender-->
    <appenders>
        <!--输出日志信息到控制台-->
        <console name="Console" target="SYSTEM_OUT">
            <!--控制日志输出的格式-->
            <PatternLayout 
				pattern="%d{yyyy-MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36} - %msg%n"/>
		</console>
    </appenders>
   
    <!--
		然后定义logger
		只有定义了logger并引入的appender,appender才会生效
	-->
    <!--
		root:用于指定项目的根日志
		如果没有单独指定Logger,则会使用root作为默认的日志输出
	-->
    <loggers>
        <root level="info">
            <appender-ref ref="Console"/>
        </root>
    </loggers>
</configuration>

3. 注解@Nullable

  • 注解用在方法上面,方法返回值可以为空
  • 注解使用在方法参数里面,方法参数可以为空
  • 注解使用在方法参数里面,方法参数可以为空

4. 函数式风格GenericApplicationContext注册bean

//函数式风格创建对象,交给spring进行管理
@Test
public void testGenericApplicationContext() {
	//1 创建GenericApplicationContext对象
	GenericApplicationContext context = new GenericApplicationContext();
	//2 调用context的方法对象注册
	context.refresh();
	context.registerBean("user1",User.class,() -> new User());
	//3 获取在spring注册的对象
    // User user = (User)context.getBean("com.my.spring5.test.User");
	User user = (User)context.getBean("user1");
	System.out.println(user);
}

5. 整合junit

  • Junit4
//单元测试框架
@RunWith(SpringJUnit4ClassRunner.class)
//加载配置文件
@ContextConfiguration("classpath:bean1.xml") 
public class JTest4 {
    
}
  • Junit5
//@ExtendWith(SpringExtension.class)
//@ContextConfiguration("classpath:bean1.xml")

//@SpringJunitConfig:一个复合注解,等价于上面二者合一
@SpringJUnitConfig(locations = "classpath:bean1.xml")
public class JTest5 {

}

WebFlux

1. SpringWebflux介绍

  • Spring5 添加新的模块,用于 web 开发的,功能和 SpringMVC 类似的, Webflux 使用当前一种比较流程响应式编程出现的框架
  • 使用传统web框架,比如SpringMVC,这些基于Servlet容器,Webflux是一种异步非阻塞的框架,异步非阻塞的框架在Servlet3.1以后才支持,核心是基于Reactor的相关API实现的

2. 异步与同步:异步和同步针对调用者

  • 调用者发送请求,如果等着对方回应之后才去做其他事情就是同步
  • 调用者发送请求,如果发送请求之后不等着对方回应就去做其他事情就是异步

3. 阻塞和非阻塞:阻塞和非阻塞针对被调用者

  • 被调用者收到请求之后,做完请求任务之后才给出反馈就是阻塞
  • 被调用者收到请求之后,马上给出反馈然后再去做事情就是非阻塞

4. WebFlux特点

  • 非阻塞式:在有限的资源下,提高系统吞吐量和伸缩性,以reactor为基础实现响应式编程
  • 函数式编程:spring5框架基于java8,webFlux使用java8函数式编程方式实现路由请求

5. SpringMvc与webFlux的比较

  • 两个框架都可以使用注解方式,都运行在tomcat等容器中
  • Spring MVC采用命令式编程,webflux采用异步响应式编程
  • Spring MVC方式实现,同步阻塞的方式,基于SpringMVC+Servlet+Tomcat
  • Spring WebFlux方式实现,异步非阻塞方式,基于SpringWebFlux+Reactor+Netty

6. 什么是响应式编程

  • 响应式编程是一种面向数据流和变化传播的编程范式。这意味着可以在编程语言中很方便地表达静态或动态的数据流,而相关的计算模型会自动将变化的值通过数据流进行传播。
  • 例如电子表格程序就是响应式编程的一个例子。单元格可以包含字面值或类似"=B1+C1"的公式,而包含公式的单元格的值会依据其他单元格的值的变化而变化。

7. 响应式编程例子(Reactor 实现)

  • 响应式编程操作中,Reactor 是满足Reactive 规范框架
  • Reactor 有两个核心类,MonoFlux,这两个类实现接口Publisher,提供丰富操作符
    • Flux对象实现发布者,返回N个元素
    • Mono实现发布者,返回0或者1个元素
  • 使用Flux和Mono都可以发出三种数据信号
    • 元素值
    • 错误信号
    • 完成信号
  • 错误信号和完成信号都代表终止信号,终止信号用于告诉订阅者数据流结束了,错误信号终止数据流同时把错误信息传递给订阅者,二者不能共存
  • 如果没有发送任何元素值,而是直接发送错误或者完成信号,表示是空数据流
  • 如果没有错误信号,也没有完成信号,表示是无限数据流
  • 操作符:对数据流进行一道道操作,成为操作符,比如工厂流水线
    • map元素映射为新元素
    • flatMap元素映射为流,把每个元素转换流,把转换之后多个流合并大的流
//调用just或者其他方法只是声明数据流,数据流并没有发出
//只有进行订阅之后才会触发数据流,不订阅什么都不会发生的
public class TestReactor {
    
	public static void main(String[] args) {
        //just方法直接声明
        Flux.just(1,2,3,4).subscribe(System.out::print);
        Mono.just(1).subscribe(System.out::print);
        
		//其他的方法
        Integer[] array = {1,2,3,4};
        Flux.fromArray(array);
        
		List<Integer> list = Arrays.asList(array);
        Flux.fromIterable(list);
        
		Stream<Integer> stream = list.stream();
        Flux.fromStream(stream);
    }
}

8. SpringWebflux 核心执行流程

  • SpringWebflux 基于 Reactor,默认使用容器是 Netty, Netty 是高性能的 NIO 框架,异步非阻塞的框架
  • SpringWebflux执行过程和SpringMVC相似
    • SpringWebflux 核心控制器DispatchHandler,实现接口 WebHandler
    • 接口WebHandler有一个方法

9. SpringWebflux 核心API

  • DispatcherHandler,负责请求的处理
  • HandlerMapping:请求查询到处理的方法
  • HandlerAdapter:真正负责请求处理
  • HandlerResultHandler:响应结果处理

10. SpringWebflux(基于函数式编程模型)

  • 在使用函数式编程模型操作时候,需要自己初始化服务器
  • 基于函数式编程模型时候,有两个核心接口:
    • RouterFunction(实现路由功能,请求转发|给对应的 handler)
    • HandlerFunction(处理请求生成响应的函数)。
    • 核心任务定义两个函数式接口的实现并且启动需要的服务器。
  • SpringWebflux请求和响应不再是ServletRequest和ServletResponse,而是ServerRequest和ServerResponse

<think>我们正在讨论的是在Spring Boot项目中如何管理`spring-boot-starter-webflux`的版本。根据用户的问题,尽管已经配置了dependencyManagement,但依赖的版本仍然是2.7.18。我们需要提供解决方案。 核心思路: 1. 确保在dependencyManagement中正确导入了Spring Boot的BOM(Bill of Materials),并且版本为期望的版本(例如3.4.5)。 2. 如果存在冲突,需要排除旧版本的传递依赖。 3. 显式声明依赖的版本(覆盖传递依赖的版本)。 根据引用[1][2][3][4],我们知道`spring-boot-starter-webflux`是Spring Boot中用于构建响应式Web应用的核心依赖,它包含了Reactor Netty、Spring WebFlux等。因此,正确管理其版本对于项目至关重要。 具体步骤: 1. **检查dependencyManagement部分**:确保导入了Spring Boot的BOM,并且版本是期望的(比如3.4.5)。注意,BOM应该放在dependencyManagement部分的最前面,以避免其他BOM覆盖其版本。 2. **使用Maven命令分析依赖树**:找出为什么2.7.18版本被引入,可能是某个传递依赖引入了旧版本的Spring Boot相关依赖。 3. **排除冲突的依赖**:在引入该传递依赖的地方,排除掉旧版本的`spring-boot-starter-webflux`或相关的Spring Boot依赖。 4. **显式声明依赖**:在dependencies部分直接指定`spring-boot-starter-webflux`的版本。 5. **检查插件版本**:确保Spring Boot Maven插件的版本与BOM版本一致。 6. **清理本地仓库**:有时候本地缓存的旧版本可能导致问题,可以清理本地仓库重新下载。 下面是一个完整的配置示例: ```xml <project> <!-- 其他配置 --> <properties> <spring-boot.version>3.4.5</spring-boot.version> </properties> <dependencyManagement> <dependencies> <!-- 1. 首先导入Spring Boot BOM --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>${spring-boot.version}</version> <type>pom</type> <scope>import</scope> </dependency> <!-- 其他依赖管理 --> </dependencies> </dependencyManagement> <dependencies> <!-- 2. 添加spring-boot-starter-webflux依赖 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-webflux</artifactId> <!-- 由于BOM已经管理版本,通常不需要指定版本,但如果冲突,可以显式指定 --> <version>${spring-boot.version}</version> </dependency> <!-- 3. 如果有其他依赖引入了旧版本的Spring Boot,需要排除 --> <!-- 示例:假设my-lib引入了旧版本的Spring Boot --> <dependency> <groupId>com.example</groupId> <artifactId>my-lib</artifactId> <version>1.0.0</version> <exclusions> <!-- 排除旧版本的spring-boot-starter-webflux --> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-webflux</artifactId> </exclusion> <!-- 或者排除整个spring-boot-starter,如果必要的话 --> <!-- <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> </exclusion> --> </exclusions> </dependency> </dependencies> <build> <plugins> <!-- 4. 确保Spring Boot Maven插件版本一致 --> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> <version>${spring-boot.version}</version> </plugin> </plugins> </build> </project> ``` 执行以下命令检查依赖树: ```bash mvn dependency:tree -Dincludes=org.springframework.boot:spring-boot-starter-webflux ``` 如果输出显示版本已经是3.4.5,则说明配置成功。如果仍有旧版本,请检查是否还有其他传递依赖引入,并进行排除。 另外,如果项目使用了Spring Cloud,请注意Spring Cloud版本与Spring Boot版本的兼容性。例如,Spring Cloud 2023.0.x需要Spring Boot 3.2.x或更高版本[^2]。 最后,清理本地Maven仓库缓存: ```bash mvn dependency:purge-local-repository -DactTransitively=false ``` 然后重新构建项目。 通过以上步骤,应该可以解决版本问题。 相关问题: 1. 如何在多模块项目中统一管理Spring Boot版本? 2. 如何升级Spring Boot 2.x到3.x? 3. WebFlux和MVC在性能上有哪些差异? 4. 如何测试WebFlux应用的响应式流? 5. 在Spring Boot中如何切换WebFlux的服务器(例如从Netty到Tomcat)?</think>### 在 Spring Boot 中管理 spring-webflux 版本的完整指南 在 Spring Boot 项目中管理 `spring-boot-starter-webflux` 版本的核心是通过 **BOM(Bill of Materials)依赖管理**实现版本统一控制。以下是具体操作步骤: --- ### 一、标准配置方法 #### 1. 使用 Spring Boot BOM 统一管理 在 `dependencyManagement` 中导入 Spring Boot 官方 BOM: ```xml <dependencyManagement> <dependencies> <!-- 核心:导入 Spring Boot BOM --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>3.4.5</version> <!-- 指定所需版本 --> <type>pom</type> <scope>import</scope> </dependency> </dependencies> </dependencyManagement> <dependencies> <!-- 无需指定版本,BOM自动管理 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-webflux</artifactId> </dependency> </dependencies> ``` 此时所有 Spring Boot 相关依赖(包括 WebFlux)将自动对齐到 3.4.5 版本[^1][^2]。 #### 2. 验证依赖版本 ```bash mvn dependency:tree -Dincludes=org.springframework.boot:spring-boot-starter-webflux ``` 期望输出: ``` [INFO] org.springframework.boot:spring-boot-starter-webflux:jar:3.4.5:compile ``` --- ### 二、版本冲突解决方案 当出现版本不匹配时(如仍显示 2.7.18): #### 1. 强制指定版本(显式覆盖) ```xml <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-webflux</artifactId> <version>3.4.5</version> <!-- 显式覆盖 --> </dependency> </dependencies> ``` #### 2. 排除冲突依赖 ```xml <dependency> <groupId>com.example</groupId> <artifactId>conflicting-lib</artifactId> <version>1.0.0</version> <exclusions> <exclusion> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-webflux</artifactId> </exclusion> </exclusions> </dependency> ``` #### 3. 检查 BOM 声明顺序 确保 Spring Boot BOM 在 `dependencyManagement` 中 **最先声明**: ```xml <dependencyManagement> <dependencies> <!-- 必须首位声明 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>3.4.5</version> <type>pom</type> <scope>import</scope> </dependency> <!-- 其他BOM(如Spring Cloud)声明在此后 --> </dependencies> </dependencyManagement> ``` --- ### 三、版本兼容性要点 | 组件 | 与 Spring Boot 3.4.5 的兼容要求 | |------------------|-------------------------------------| | **Reactor Netty** | 自动由 BOM 管理(默认服务器容器)[^1] | | **Spring Security** | 需使用 6.1.x+ 版本 | | **Spring Data** | 需使用 2023.1.x+ 版本 | | **Jakarta EE** | 必须使用 Jakarta EE 9+ API | | **JDK** | 最低要求 JDK 17 | > 提示:WebFlux 保留 Spring MVC 的注解风格(如 `@RestController`),但底层使用响应式编程模型[^2][^3]。 --- ### 四、验证配置 #### 1. 检查实际依赖版本 ```bash mvn dependency:list -DincludeArtifactIds=spring-boot-starter-webflux ``` #### 2. 创建测试端点 ```java @RestController public class DemoController { @GetMapping("/latency") public Mono<String> testLatency() { return Mono.just("WebFlux response in: " + System.currentTimeMillis()); } } ``` 启动应用后访问 `http://localhost:8080/latency`,观察响应式行为[^4]。 --- ### 五、升级注意事项 1. **包名变化**: Spring Boot 3 使用 Jakarta EE 9+ 的 `jakarta.servlet` 替代 `javax.servlet` 2. **响应式数据库**: 需配合响应式驱动(如 R2DBC): ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-r2dbc</artifactId> </dependency> ``` 3. **安全配置**: Spring Security 需使用响应式配置: ```java @EnableWebFluxSecurity public class SecurityConfig { @Bean public SecurityWebFilterChain securityWebFilterChain(ServerHttpSecurity http) { return http.authorizeExchange().anyExchange().authenticated().and().build(); } } ``` --- ### 相关问题 1. 如何将 Spring Boot 2.x 的 WebFlux 应用升级到 3.x 版本? 2. WebFlux 和 MVC 在性能上有哪些具体差异?[^4] 3. 响应式编程中如何处理阻塞操作? 4. Spring Security 在 WebFlux 中的配置与 MVC 有何不同?[^2] 5. 如何监控 WebFlux 应用的响应式流性能? > 通过 BOM 管理可确保 Spring 生态组件的版本兼容性,避免依赖冲突[^1][^2]。若需覆盖版本,应在排除冲突依赖后显式指定目标版本。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值