跳转到主要内容

标签(标签)

资源精选(342) Go开发(108) Go语言(103) Go(99) angular(83) LLM(78) 大语言模型(63) 人工智能(53) 前端开发(50) LangChain(43) golang(43) 机器学习(39) Go工程师(38) Go程序员(38) Go开发者(36) React(34) Go基础(29) Python(24) Vue(23) Web开发(20) Web技术(19) 精选资源(19) 深度学习(19) Java(18) ChatGTP(17) Cookie(16) android(16) 前端框架(13) JavaScript(13) Next.js(12) 安卓(11) 聊天机器人(10) typescript(10) 资料精选(10) NLP(10) 第三方Cookie(9) Redwoodjs(9) ChatGPT(9) LLMOps(9) Go语言中级开发(9) 自然语言处理(9) PostgreSQL(9) 区块链(9) mlops(9) 安全(9) 全栈开发(8) OpenAI(8) Linux(8) AI(8) GraphQL(8) iOS(8) 软件架构(7) RAG(7) Go语言高级开发(7) AWS(7) C++(7) 数据科学(7) 智能体(6) whisper(6) Prisma(6) 隐私保护(6) JSON(6) DevOps(6) 数据可视化(6) wasm(6) 计算机视觉(6) 算法(6) Rust(6) 微服务(6) 隐私沙盒(5) FedCM(5) 语音识别(5) Angular开发(5) 快速应用开发(5) 提示工程(5) Agent(5) LLaMA(5) 低代码开发(5) Go测试(5) gorm(5) REST API(5) kafka(5) 推荐系统(5) WebAssembly(5) GameDev(5) CMS(5) CSS(5) machine-learning(5) 机器人(5) 游戏开发(5) Blockchain(5) Web安全(5) nextjs(5) Kotlin(5) 低代码平台(5) 机器学习资源(5) Go资源(5) Nodejs(5) PHP(5) Swift(5) devin(4) Blitz(4) javascript框架(4) Redwood(4) GDPR(4) 生成式人工智能(4) Angular16(4) Alpaca(4) 编程语言(4) SAML(4) JWT(4) JSON处理(4) Go并发(4) 移动开发(4) 移动应用(4) security(4) 隐私(4) spring-boot(4) 物联网(4) 网络安全(4) API(4) Ruby(4) 信息安全(4) flutter(4) RAG架构(3) 专家智能体(3) Chrome(3) CHIPS(3) 3PC(3) SSE(3) 人工智能软件工程师(3) LLM Agent(3) Remix(3) Ubuntu(3) GPT4All(3) 软件开发(3) 问答系统(3) 开发工具(3) 最佳实践(3) RxJS(3) SSR(3) Node.js(3) Dolly(3) 移动应用开发(3) 低代码(3) IAM(3) Web框架(3) CORS(3) 基准测试(3) Go语言数据库开发(3) Oauth2(3) 并发(3) 主题(3) Theme(3) earth(3) nginx(3) 软件工程(3) azure(3) keycloak(3) 生产力工具(3) gpt3(3) 工作流(3) C(3) jupyter(3) 认证(3) prometheus(3) GAN(3) Spring(3) 逆向工程(3) 应用安全(3) Docker(3) Django(3) R(3) .NET(3) 大数据(3) Hacking(3) 渗透测试(3) C++资源(3) Mac(3) 微信小程序(3) Python资源(3) JHipster(3) 语言模型(2) 可穿戴设备(2) JDK(2) SQL(2) Apache(2) Hashicorp Vault(2) Spring Cloud Vault(2) Go语言Web开发(2) Go测试工程师(2) WebSocket(2) 容器化(2) AES(2) 加密(2) 输入验证(2) ORM(2) Fiber(2) Postgres(2) Gorilla Mux(2) Go数据库开发(2) 模块(2) 泛型(2) 指针(2) HTTP(2) PostgreSQL开发(2) Vault(2) K8s(2) Spring boot(2) R语言(2) 深度学习资源(2) 半监督学习(2) semi-supervised-learning(2) architecture(2) 普罗米修斯(2) 嵌入模型(2) productivity(2) 编码(2) Qt(2) 前端(2) Rust语言(2) NeRF(2) 神经辐射场(2) 元宇宙(2) CPP(2) 数据分析(2) spark(2) 流处理(2) Ionic(2) 人体姿势估计(2) human-pose-estimation(2) 视频处理(2) deep-learning(2) kotlin语言(2) kotlin开发(2) burp(2) Chatbot(2) npm(2) quantum(2) OCR(2) 游戏(2) game(2) 内容管理系统(2) MySQL(2) python-books(2) pentest(2) opengl(2) IDE(2) 漏洞赏金(2) Web(2) 知识图谱(2) PyTorch(2) 数据库(2) reverse-engineering(2) 数据工程(2) swift开发(2) rest(2) robotics(2) ios-animation(2) 知识蒸馏(2) 安卓开发(2) nestjs(2) solidity(2) 爬虫(2) 面试(2) 容器(2) C++精选(2) 人工智能资源(2) Machine Learning(2) 备忘单(2) 编程书籍(2) angular资源(2) 速查表(2) cheatsheets(2) SecOps(2) mlops资源(2) R资源(2) DDD(2) 架构设计模式(2) 量化(2) Hacking资源(2) 强化学习(2) flask(2) 设计(2) 性能(2) Sysadmin(2) 系统管理员(2) Java资源(2) 机器学习精选(2) android资源(2) android-UI(2) Mac资源(2) iOS资源(2) Vue资源(2) flutter资源(2) JavaScript精选(2) JavaScript资源(2) Rust开发(2) deeplearning(2) RAD(2)

category

1.概述


在本教程中,我们将看到如何使用Spring实现基于服务器发送事件的API。

简单地说,Server Sent Events(简称SSE)是一种HTTP标准,它允许web应用程序处理单向事件流,并在服务器发出数据时接收更新。

Spring4.2版本已经支持它,但从Spring5开始,我们现在有了一种更惯用、更方便的方法来处理它。

2.带Spring 6 Webflux的SSE


为了实现这一点,我们可以使用Reactor库提供的Flux类等实现,或者潜在的ServerSentEvent实体,它使我们能够控制事件元数据。

2.1. 使用通量的流事件


Flux是事件流的反应性表示——它根据指定的请求或响应媒体类型进行不同的处理。

要创建SSE流式端点,我们必须遵循W3C规范,并将其MIME类型指定为文本/事件流:

@GetMapping(path = "/stream-flux", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
public Flux<String> streamFlux() {
    return Flux.interval(Duration.ofSeconds(1))
      .map(sequence -> "Flux - " + LocalTime.now().toString());
}


interval方法创建一个以增量方式发射长值的Flux。然后我们将这些值映射到我们想要的输出。

让我们启动我们的应用程序,然后通过浏览端点来尝试它。

我们将看到浏览器对服务器逐秒推送的事件的反应。有关通量和堆芯的更多信息,我们可以查看这篇文章。

2.2. 使用ServerSentEvent元素


现在,我们将把输出字符串包装到ServerSentSevent对象中,并检查这样做的好处:
@GetMapping("/stream-sse")
public Flux<ServerSentEvent<String>> streamEvents() {
    return Flux.interval(Duration.ofSeconds(1))
      .map(sequence -> ServerSentEvent.<String> builder()
        .id(String.valueOf(sequence))
          .event("periodic-event")
          .data("SSE - " + LocalTime.now().toString())
          .build());
}


正如我们所理解的,使用ServerSentEvent实体有几个好处:

  • 我们可以处理事件元数据,这在实际情况下是需要的
  • 我们可以忽略“文本/事件流”媒体类型声明

在这种情况下,我们指定了一个id、一个事件名称,最重要的是,指定了事件的实际数据。

此外,我们还可以添加一个comments属性和一个retry值,该值将指定尝试发送事件时要使用的重新连接时间。

2.3. 使用WebClient使用服务器发送的事件


现在,让我们使用WebClient来使用我们的事件流。:

public void consumeServerSentEvent() {
    WebClient client = WebClient.create("http://localhost:8080/sse-server");
    ParameterizedTypeReference<ServerSentEvent<String>> type
     = new ParameterizedTypeReference<ServerSentEvent<String>>() {};

    Flux<ServerSentEvent<String>> eventStream = client.get()
      .uri("/stream-sse")
      .retrieve()
      .bodyToFlux(type);

    eventStream.subscribe(
      content -> logger.info("Time: {} - event: name[{}], id [{}], content[{}] ",
        LocalTime.now(), content.event(), content.id(), content.data()),
      error -> logger.error("Error receiving SSE: {}", error),
      () -> logger.info("Completed!!!"));
}

 

订阅方法允许我们指示当我们成功接收到事件、发生错误以及流传输完成时将如何进行。

在我们的示例中,我们使用了retrieve方法,这是一种获取响应体的简单而直接的方法。

如果我们收到4xx或5xx响应,除非我们处理添加onStatus语句的场景,否则此方法会自动抛出WebClientResponseException。

另一方面,我们也可以使用exchange方法,该方法提供对ClientResponse的访问,并且在失败响应时不会发出错误信号。

我们必须考虑到,如果不需要事件元数据,我们可以绕过ServerSentEvent包装器。

3.Spring MVC中的SSE流
正如我们所说,SSE规范自引入SseEmitter类的Spring4.2以来就得到了支持。

简单地说,我们将定义一个ExecutorService,一个SseEmitter将在其中推送数据的线程,并返回发射器实例,以这种方式保持连接打开:

@GetMapping("/stream-sse-mvc")
public SseEmitter streamSseMvc() {
    SseEmitter emitter = new SseEmitter();
    ExecutorService sseMvcExecutor = Executors.newSingleThreadExecutor();
    sseMvcExecutor.execute(() -> {
        try {
            for (int i = 0; true; i++) {
                SseEventBuilder event = SseEmitter.event()
                  .data("SSE MVC - " + LocalTime.now().toString())
                  .id(String.valueOf(i))
                  .name("sse event - mvc");
                emitter.send(event);
                Thread.sleep(1000);
            }
        } catch (Exception ex) {
            emitter.completeWithError(ex);
        }
    });
    return emitter;
}


始终确保为您的用例场景选择正确的ExecutiorService。

我们可以在SpringMVC中了解更多关于SSE的信息,并通过阅读这篇有趣的教程来了解其他示例。

4.了解服务器发送的事件


现在我们知道了如何实现SSE端点,让我们通过理解一些基本概念来尝试更深入地了解一下。

SSE是大多数浏览器采用的一种规范,允许在任何时候单向地流式传输事件。

“事件”只是一个UTF-8编码的文本数据流,遵循规范定义的格式。

此格式由一系列用换行符分隔的键值元素(id、retry、data和event,表示名称)组成。

也支持评论。

该规范没有以任何方式限制数据有效载荷格式;我们可以使用简单的String或更复杂的JSON或XML结构。

我们必须考虑的最后一点是使用SSE流和WebSockets之间的区别。

WebSockets提供服务器和客户端之间的全双工(双向)通信,而SSE使用单向通信。
此外,WebSockets不是HTTP协议,与SSE相反,它不提供错误处理标准。

5.结论


总之,在本文中,我们学习了SSE流的主要概念,这无疑是一个很好的资源,可以让我们创建下一代系统。

当我们使用该协议时,我们现在处于一个很好的位置来了解幕后发生的事情。

此外,我们还用一些简单的例子补充了这一理论,这些例子可以在我们的Github存储库中找到。
 

标签