0

0

Spring WebFlux中响应式流处理null值与自定义异常抛出

心靈之曲

心靈之曲

发布时间:2025-08-23 17:50:41

|

262人浏览过

|

来源于php中文网

原创

Spring WebFlux中响应式流处理null值与自定义异常抛出

本文旨在探讨Spring WebFlux响应式编程中,如何优雅地处理响应式流中可能出现的null值,并在检测到null时抛出自定义异常。我们将深入分析为什么直接使用map结合switchIfEmpty或filter无法达到预期效果,并提供两种推荐的解决方案:flatMap和handle操作符,以确保响应式流的健壮性和错误处理的准确性。

理解响应式流中的null值限制

在project reactor(spring webflux底层使用的响应式库)以及更广泛的reactive streams规范中,null值是被明确禁止作为流中元素的。这意味着,一个mono或flux不应该发出null值。当一个操作符(如map)的映射函数返回null时,reactor会将其视为一个无效的信号,并可能导致意外行为或运行时错误。因此,在响应式编程中,我们不能简单地让一个map操作返回null,然后期望后续的操作符能像处理空流一样处理它。

问题场景:条件性抛出异常

考虑以下Spring WebFlux中的响应式代码片段,其目标是从安全上下文中获取用户主体名称(userPrincipalName):

ReactiveSecurityContextHolder.getContext()
        .map(SecurityContext::getAuthentication)
        .map(authentication -> (UserAuthenticationToken) authentication)
        .map(UserAuthenticationToken::getUserPrincipalName)
        // 在这里,如果getUserPrincipalName返回null,我们希望抛出自定义异常

如果UserAuthenticationToken::getUserPrincipalName方法可能返回null,并且我们希望在这种情况下抛出一个自定义的MissingPrincipalException,那么直接尝试使用switchIfEmpty或filter是无效的:

// 尝试1:使用switchIfEmpty - 无效
.map(UserAuthenticationToken::getUserPrincipalName)
.switchIfEmpty(Mono.error(new MissingPrincipalException("Missing email field in the JWT token")));

// 尝试2:使用filter结合switchIfEmpty - 无效
.map(UserAuthenticationToken::getUserPrincipalName)
.filter(Objects::nonNull) // 过滤掉null
.switchIfEmpty(Mono.error(new MissingPrincipalException("Missing email field in the JWT token")));

这些尝试之所以无效,是因为switchIfEmpty操作符是用来处理上游Publisher发出完成信号且未发出任何元素(即流为空)的情况。当map操作返回null时,它仍然会尝试将null作为元素发出,这违反了Reactive Streams规范。filter(Objects::nonNull)虽然可以过滤掉null,但如果map本身已经发出了null,在filter之前就已经违反了规范。更关键的是,即使filter成功过滤了null,switchIfEmpty也只会在整个流变为空时才触发,而不是在单个元素为null时触发。

解决方案一:使用flatMap操作符

flatMap操作符允许我们将流中的每个元素映射到一个新的Publisher(Mono或Flux)。这使得我们可以在映射过程中执行条件逻辑,并根据条件返回不同的Publisher,包括一个错误Publisher。

示例代码:

import org.springframework.security.core.context.ReactiveSecurityContextHolder;
import org.springframework.security.core.context.SecurityContext;
import reactor.core.publisher.Mono;

// 假设UserAuthenticationToken和MissingPrincipalException已定义
// class UserAuthenticationToken implements Authentication { ... public String getUserPrincipalName() { ... } }
// class MissingPrincipalException extends RuntimeException { ... }

public class PrincipalService {

    public Mono getUserPrincipalNameSafely() {
        return ReactiveSecurityContextHolder.getContext()
                .map(SecurityContext::getAuthentication)
                .map(authentication -> (UserAuthenticationToken) authentication)
                .flatMap(token -> {
                    String principalName = token.getUserPrincipalName();
                    if (principalName == null) {
                        // 如果principalName为null,则返回一个包含错误的Mono
                        return Mono.error(new MissingPrincipalException("Missing email field in the JWT token"));
                    } else {
                        // 否则,返回一个包含非null principalName的Mono
                        return Mono.just(principalName);
                    }
                });
    }
}

代码解析:

  1. flatMap(token -> { ... }):对于上游发出的每个UserAuthenticationToken,flatMap的函数体会被调用。
  2. 在函数体内,我们首先获取principalName。
  3. if (principalName == null):如果principalName为null,我们通过Mono.error(new MissingPrincipalException(...))创建一个包含自定义异常的Mono并返回。这个错误Mono会立即向下游发出错误信号,终止当前流。
  4. else { return Mono.just(principalName); }:如果principalName非null,我们通过Mono.just(principalName)创建一个包含该值的Mono并返回。这个Mono会正常向下游发出principalName。

flatMap的这种用法完美地解决了在特定条件(值null)下抛出异常的需求,因为它允许我们完全控制每个元素如何被转换为后续的响应式流。

FreeTTS
FreeTTS

FreeTTS是一个免费开源的在线文本到语音生成解决方案,可以将文本转换成MP3,

下载

解决方案二:使用handle操作符

handle操作符是一个功能非常强大的通用操作符,它结合了map和filter的功能,并且允许更复杂的条件逻辑,包括发出错误信号。它接收一个BiConsumer,其中包含当前元素和SynchronousSink,通过SynchronousSink可以发出零个、一个或一个错误信号。

示例代码:

import org.springframework.security.core.context.ReactiveSecurityContextHolder;
import org.springframework.security.core.context.SecurityContext;
import reactor.core.publisher.Mono;
import reactor.core.publisher.SynchronousSink;

// 假设UserAuthenticationToken和MissingPrincipalException已定义

public class PrincipalService {

    public Mono getUserPrincipalNameSafelyWithHandle() {
        return ReactiveSecurityContextHolder.getContext()
                .map(SecurityContext::getAuthentication)
                .map(authentication -> (UserAuthenticationToken) authentication)
                .handle((token, sink) -> {
                    String principalName = token.getUserPrincipalName();
                    if (principalName == null) {
                        // 如果principalName为null,通过sink发出错误信号
                        sink.error(new MissingPrincipalException("Missing email field in the JWT token"));
                    } else {
                        // 否则,通过sink发出非null principalName
                        sink.next(principalName);
                    }
                });
    }
}

代码解析:

  1. handle((token, sink) -> { ... }):对于上游发出的每个UserAuthenticationToken,handle的函数体会被调用。
  2. 在函数体内,我们获取principalName。
  3. if (principalName == null):如果principalName为null,我们调用sink.error(new MissingPrincipalException(...))发出错误信号。这会立即终止流并向下游传递异常。
  4. else { sink.next(principalName); }:如果principalName非null,我们调用sink.next(principalName)发出这个非null的值,流会继续正常处理。

handle操作符的优势在于其灵活性。它不仅可以用于条件性地发出错误,还可以用于过滤元素(不调用sink.next)、转换元素(调用sink.next一次)甚至将一个元素转换为零个或多个元素(尽管对于单值Mono来说,通常是零个或一个)。

总结与最佳实践

在Spring WebFlux响应式编程中处理可能为null的值并抛出自定义异常时,请记住以下几点:

  1. Reactive Streams规范禁止null值:不要期望map操作返回null后能被后续操作符正常处理。
  2. flatMap是处理条件性错误和转换的直接方式:当你需要根据元素的值来决定是继续传递一个值、传递一个空流还是传递一个错误流时,flatMap是首选。它将一个元素转换为一个新的Publisher。
  3. handle提供更细粒度的控制:handle操作符非常强大,适用于更复杂的条件逻辑,包括过滤、转换和发出错误。它通过SynchronousSink提供了直接控制流发出的能力。
  4. 避免switchIfEmpty和filter的误用:这些操作符主要用于处理流的空性,而不是流中单个元素的null值。
  5. 自定义异常的合理使用:针对特定的业务错误场景定义和抛出自定义异常,有助于提高代码的可读性和错误处理的准确性。

根据具体的业务逻辑和代码的简洁性需求,可以选择flatMap或handle来实现对null值的健壮处理和异常抛出。在大多数需要根据元素内容进行条件性错误处理的场景中,flatMap通常是更直观和简洁的选择。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

107

2025.08.06

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

233

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

437

2024.03.01

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

760

2023.08.22

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

188

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

288

2023.10.25

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6106

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

812

2023.09.14

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

4

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号