深入理解 `Sinks.Empty<Void>` 和 `Mono<Void>`:如何触发完成信号并结合 WebSocket 示例

server/2025/2/26 22:50:38/

在响应式编程中,Sinks 是 Project Reactor 提供的一个强大工具,用于手动控制数据流的信号发射。Sinks.Empty<Void> 是一种特殊的 Sinks,它不发射任何数据,仅用于表示完成或错误信号。结合 Mono<Void>,它可以用来表示一个异步操作的完成状态。本文将详细分析 Sinks.Empty<Void>Mono<Void> 的行为,并通过示例代码展示其用法,同时结合 WebSocket 建立连接的伪代码,展示其在实际场景中的应用。


1. Sinks.Empty<Void> 是什么?

Sinks.Empty<Void> 是 Project Reactor 中的一个 Sinks 类型,专门用于表示一个不发射任何数据的信号源。它的主要特点是:

  • 不发射任何数据(onNext 信号)。
  • 只能发射完成信号(onComplete)或错误信号(onError)。
  • 需要通过显式调用方法(如 tryEmitEmpty())来触发完成信号。

它的典型使用场景是表示一个异步操作的完成状态,而不需要传递任何数据。


2. Mono<Void> 的作用

Mono<Void> 是 Project Reactor 中的一个响应式类型,表示一个最多发射一个元素的异步序列。对于 Mono<Void> 来说:

  • 它不会发射任何数据(onNext 信号)。
  • 它只会发射完成信号(onComplete)或错误信号(onError)。
  • 它通常用于表示一个不需要返回值的异步操作。

通过将 Sinks.Empty<Void> 转换为 Mono<Void>,我们可以将手动控制的信号源与响应式流结合起来。


3. 如何触发完成信号?

Sinks.Empty<Void> 的完成信号需要通过显式调用 tryEmitEmpty() 来触发。以下是关键点:

  • 在调用 tryEmitEmpty() 之前,Sinks.Empty<Void> 处于未完成状态,订阅者会一直等待。
  • 调用 tryEmitEmpty() 后,Sinks.Empty<Void> 会立即发出完成信号,订阅者会收到 onComplete 通知。

4. 示例代码

以下是一个完整的示例,展示了如何使用 Sinks.Empty<Void>Mono<Void> 来触发完成信号:

import reactor.core.publisher.Mono;
import reactor.core.publisher.Sinks;public class SinksEmptyExample {public static void main(String[] args) {// 1. 创建一个空的 Sinks.Empty<Void>Sinks.Empty<Void> completion = Sinks.empty();// 2. 将其转换为 Mono<Void>Mono<Void> mono = completion.asMono();// 3. 订阅 Mono<Void>mono.subscribe(null, // onNext (不会调用,因为没有数据)error -> System.err.println("Error: " + error), // onError() -> System.out.println("Completed!") // onComplete);// 4. 模拟一个异步操作System.out.println("Starting async operation...");try {Thread.sleep(2000); // 模拟耗时操作} catch (InterruptedException e) {e.printStackTrace();}// 5. 手动触发完成信号System.out.println("Triggering completion...");completion.tryEmitEmpty();// 输出:// Starting async operation...// (等待 2 秒)// Triggering completion...// Completed!}
}
代码解析:
  1. 创建 Sinks.Empty<Void>

    • 使用 Sinks.empty() 创建一个空的 Sinks.Empty<Void>
  2. 转换为 Mono<Void>

    • 通过 completion.asMono()Sinks.Empty<Void> 转换为 Mono<Void>
  3. 订阅 Mono<Void>

    • 订阅 Mono<Void>,并定义 onErroronComplete 的处理逻辑。
  4. 模拟异步操作

    • 使用 Thread.sleep(2000) 模拟一个耗时 2 秒的异步操作。
  5. 触发完成信号

    • 调用 completion.tryEmitEmpty() 手动触发完成信号,订阅者会收到 onComplete 通知。

5. 结合 WebSocket 建立连接的伪代码

在实际应用中,Sinks.Empty<Void>Mono<Void> 可以用于表示 WebSocket 连接的建立和关闭。以下是一个伪代码示例,展示如何在 WebSocket 连接建立后触发完成信号:

import reactor.core.publisher.Mono;
import reactor.core.publisher.Sinks;
import org.springframework.web.reactive.socket.WebSocketSession;public class WebSocketExample {public Mono<Void> handleWebSocketConnection(WebSocketSession session) {// 1. 创建一个空的 Sinks.Empty<Void>Sinks.Empty<Void> completion = Sinks.empty();// 2. 将其转换为 Mono<Void>Mono<Void> mono = completion.asMono();// 3. 处理 WebSocket 连接session.receive().doOnNext(message -> {// 处理收到的消息System.out.println("Received message: " + message.getPayloadAsText());}).doOnError(error -> {// 处理错误System.err.println("WebSocket error: " + error);completion.tryEmitError(error); // 触发错误信号}).doOnComplete(() -> {// 连接关闭时触发完成信号System.out.println("WebSocket connection closed.");completion.tryEmitEmpty(); // 触发完成信号}).subscribe();// 4. 返回 Mono<Void>,表示 WebSocket 连接的处理结果return mono;}public static void main(String[] args) {WebSocketExample example = new WebSocketExample();// 模拟 WebSocket 连接WebSocketSession session = // 获取 WebSocketSession 的伪代码example.handleWebSocketConnection(session).subscribe(null, // onNext (不会调用,因为没有数据)error -> System.err.println("Error: " + error), // onError() -> System.out.println("WebSocket handling completed!") // onComplete);}
}
伪代码解析:
  1. 创建 Sinks.Empty<Void>

    • 使用 Sinks.empty() 创建一个空的 Sinks.Empty<Void>,用于表示 WebSocket 连接的完成状态。
  2. 处理 WebSocket 连接

    • 使用 session.receive() 处理 WebSocket 消息。
    • doOnNext 中处理收到的消息。
    • doOnError 中处理错误,并调用 completion.tryEmitError(error) 触发错误信号。
    • doOnComplete 中处理连接关闭,并调用 completion.tryEmitEmpty() 触发完成信号。
  3. 返回 Mono<Void>

    • 返回 Mono<Void>,表示 WebSocket 连接的处理结果。
  4. 订阅 Mono<Void>

    • 订阅 Mono<Void>,并定义 onErroronComplete 的处理逻辑。

6. 关键点总结

  • Sinks.Empty<Void> 是一个手动控制的信号源,它不会自动发出完成信号,必须通过显式调用 tryEmitEmpty() 来触发。
  • Mono<Void> 表示一个不发射数据的异步序列,它只会发出完成或错误信号。
  • WebSocket 连接示例
    • 通过 Sinks.Empty<Void>Mono<Void>,可以灵活地表示 WebSocket 连接的完成状态。
    • 在连接关闭或发生错误时,手动触发完成或错误信号。

7. 总结

通过 Sinks.Empty<Void>Mono<Void>,我们可以灵活地控制异步操作的完成信号。结合 WebSocket 示例,展示了如何在实际场景中使用这些工具。关键在于理解 Sinks.Empty<Void> 的初始状态是未完成的,必须通过显式调用 tryEmitEmpty()tryEmitError() 来触发信号。


http://www.ppmy.cn/server/170863.html

相关文章

【Python量化金融实战】-第1章:Python量化金融概述:1.1量化金融的定义与发展历程

本小节学习建议&#xff1a;掌握Python编程、统计学&#xff08;时间序列分析&#xff09;、金融学基础&#xff08;资产定价理论&#xff09;三者结合&#xff0c;是进入量化领域的核心路径。 &#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 文章目录 1.1 量化金…

Python 的 inspect 模块中的Signature 对象

Signature 对象是 Python 的 inspect 模块中用来描述可调用对象&#xff08;例如函数&#xff09;的“调用签名”的一种数据结构。换句话说&#xff0c;它是一种结构化的表示形式&#xff0c;告诉你一个函数需要哪些参数、参数的类型、默认值以及函数的返回注解等信息。 主要特…

【UCB CS 61B SP24】Lecture 14 - Data Structures 1: Disjoint Sets学习笔记

本文内容为数据结构并查集&#xff08;DSU&#xff09;的介绍与实现&#xff0c;详细讲解了并查集这一数据结构所能实现的各种操作&#xff0c;以及如何通过路径压缩与按秩合并大幅优化并查集的效率。 1. 并查集 1.1 介绍及其基础操作 并查集&#xff08;Disjoint Set Union…

STM32的HAL库开发---单通道ADC采集(DMA读取)实验

一、实验简介 正常单通道ADC采集顺序是先开启ADC采集&#xff0c;然后等待ADC转换完成&#xff0c;也就是判断EOC位置1&#xff0c;然后再读取数据寄存器的值。 如果配置了DMA功能&#xff0c;在EOC位被硬件置1后&#xff0c;自动产生DMA请求&#xff0c;然后DMA进行数据搬运…

机器翻译与语音识别技术:推动人机交互的新篇章

在数字化时代&#xff0c;语言不仅是人类交流的基本工具&#xff0c;也是连接不同文化和国家的桥梁。随着科技的飞速发展&#xff0c;机器翻译与语音识别技术作为语言处理领域的两大核心技术&#xff0c;正逐步改变着人类与计算机之间的交互方式。本文将深入探讨这两种技术的原…

图片爬取案例

修改前的代码 但是总显示“失败” 原因是 修改之后的代码 import requests import os from urllib.parse import unquote# 原始URL url https://cn.bing.com/images/search?viewdetailV2&ccidTnImuvQ0&id5AE65CE4BE05EE7A79A73EEFA37578E87AE19421&thidOIP.TnI…

危化品经营单位安全管理人员的职责及注意事项

危化品经营单位安全管理人员肩负着保障经营活动安全的重要责任&#xff0c;以下是其主要职责及注意事项&#xff1a; 职责 1. 安全制度建设与执行&#xff1a;负责组织制定本单位安全生产规章制度、操作规程和生产安全事故应急救援预案&#xff0c;确保这些制度符合国家相关法…

ArcGIS Pro中生成带计曲线等高线的全面指南

一、引言 在地理信息系统&#xff08;GIS&#xff09;领域&#xff0c;等高线作为表达地形起伏的重要视觉元素&#xff0c;被广泛应用于地图制作、空间分析以及地形可视化等方面。ArcGIS Pro&#xff0c;作为Esri公司推出的新一代GIS平台&#xff0c;提供了强大的空间分析和地…