当前位置:   article > 正文

打造聊天流式回复效果:Spring Boot+WebSocket + JS实战_webclient 流式返回到前端

webclient 流式返回到前端

本篇博客将带领你使用 Spring Boot、WebSocket 和 JavaScript 实现一个类似 ChatGPT 的流式回复效果。前端发送消息后,后端接收消息并请求 AI API,并将 AI 返回的流式响应实时推送到前端,最终在聊天界面呈现出逐字出现的打字效果。

技术原理

  1. WebSocket 全双工通信: WebSocket 协议提供全双工通信通道,允许服务器和客户端之间进行双向实时数据传输,非常适合实现流式数据传输。

  2. Spring WebFlux 响应式编程: Spring WebFlux 基于 Reactor 库,支持响应式流处理,可以处理 AI API 返回的流式数据。

  3. SSE(Server-Sent Events): SSE 是一种基于 HTTP 的单向服务器推送技术,可以将 AI 返回的流式数据实时推送到前端。

实现步骤

一、后端实现 (Spring Boot)
  1. 添加依赖
  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-websocket</artifactId>
  4. </dependency>
  5. <dependency>
  6. <groupId>org.springframework.boot</groupId>
  7. <artifactId>spring-boot-starter-webflux</artifactId>
  8. </dependency>
  1. 创建 WebSocket 处理器
  1. import org.springframework.beans.factory.annotation.Autowired;
  2. import org.springframework.messaging.handler.annotation.MessageMapping;
  3. import org.springframework.messaging.handler.annotation.Payload;
  4. import org.springframework.messaging.simp.SimpMessagingTemplate;
  5. import org.springframework.stereotype.Controller;
  6. import org.springframework.web.reactive.function.client.WebClient;
  7. import reactor.core.publisher.Flux;
  8. import java.time.Duration;
  9. @Controller
  10. public class ChatController {
  11. @Autowired
  12. private SimpMessagingTemplate messagingTemplate;
  13. @Autowired
  14. private WebClient webClient; // 用于调用 AI API
  15. @MessageMapping("/chat.sendMessage") // 接收来自客户端的消息
  16. public void sendMessage(@Payload Message message) throws Exception {
  17. // 构造 AI API 请求
  18. ApiRequest apiRequest = new ApiRequest(message.getContent()); // 假设 AI API 接受 ApiRequest 对象
  19. // 调用 AI API,获取流式响应
  20. Flux<String> apiResponse = callAiApi(apiRequest);
  21. // 处理 AI API 响应,并逐条发送给前端
  22. apiResponse.delayElements(Duration.ofMillis(50)) // 模拟打字延迟
  23. .subscribe(chunk -> {
  24. Message responseMessage = new Message("AI", chunk);
  25. messagingTemplate.convertAndSend("/topic/chat", responseMessage);
  26. });
  27. }
  28. // 调用 AI API
  29. private Flux<String> callAiApi(ApiRequest request) {
  30. // 将 ApiRequest 对象转换为 JSON 字符串
  31. // ...
  32. // 发送 POST 请求,并指定返回数据类型为 Flux<String>
  33. return webClient.post()
  34. .uri("https://api.example.com/ai-api") // 替换为实际的 AI API 地址
  35. .bodyValue(request)
  36. .retrieve()
  37. .bodyToFlux(String.class);
  38. }
  39. // 消息类
  40. public static class Message {
  41. private String sender;
  42. private String content;
  43. // 构造函数,getter 和 setter
  44. }
  45. // AI API 请求类
  46. public static class ApiRequest {
  47. private String prompt;
  48. // 构造函数,getter 和 setter
  49. }
  50. }
  1. WebSocket 配置
  1. import org.springframework.context.annotation.Configuration;
  2. import org.springframework.messaging.simp.config.MessageBrokerRegistry;
  3. import org.springframework.web.socket.config.annotation.EnableWebSocketMessageBroker;
  4. import org.springframework.web.socket.config.annotation.StompEndpointRegistry;
  5. import org.springframework.web.socket.config.annotation.WebSocketMessageBrokerConfigurer;
  6. @Configuration
  7. @EnableWebSocketMessageBroker
  8. public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {
  9. @Override
  10. public void configureMessageBroker(MessageBrokerRegistry config) {
  11. config.enableSimpleBroker("/topic");
  12. config.setApplicationDestinationPrefixes("/app");
  13. }
  14. @Override
  15. public void registerStompEndpoints(StompEndpointRegistry registry) {
  16. registry.addEndpoint("/ws").withSockJS();
  17. }
  18. }
二、前端实现 (JavaScript)
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <title>Chat Application</title>
  5. <script src="https://cdnjs.cloudflare.com/ajax/libs/sockjs-client/1.5.1/sockjs.min.js"></script>
  6. <script src="https://cdnjs.cloudflare.com/ajax/libs/stomp.js/2.3.3/stomp.min.js"></script>
  7. </head>
  8. <body>
  9. <div id="chat-container">
  10. <div id="chat-output"></div>
  11. <input type="text" id="message-input" placeholder="Enter message...">
  12. <button onclick="sendMessage()">Send</button>
  13. </div>
  14. <script>
  15. // 连接 WebSocket
  16. var socket = new SockJS('/ws');
  17. var stompClient = Stomp.over(socket);
  18. stompClient.connect({}, function(frame) {
  19. console.log('Connected: ' + frame);
  20. // 订阅聊天频道
  21. stompClient.subscribe('/topic/chat', function(message) {
  22. showMessage(JSON.parse(message.body));
  23. });
  24. });
  25. // 发送消息
  26. function sendMessage() {
  27. var message = document.getElementById('message-input').value;
  28. stompClient.send("/app/chat.sendMessage", {}, JSON.stringify({
  29. 'sender': 'user', // 可替换为实际用户名
  30. 'content': message
  31. }));
  32. }
  33. // 显示消息
  34. function showMessage(message) {
  35. var chatOutput = document.getElementById('chat-output');
  36. var messageElement = document.createElement('div');
  37. if (message.sender === 'AI') {
  38. // AI 回复逐字显示
  39. typeWriter(messageElement, message.content, 0);
  40. } else {
  41. messageElement.innerHTML = '<strong>' + message.sender + ':</strong> ' + message.content;
  42. chatOutput.appendChild(messageElement);
  43. }
  44. }
  45. // 模拟打字效果
  46. function typeWriter(element, text, i) {
  47. if (i < text.length) {
  48. element.innerHTML += text.charAt(i);
  49. setTimeout(function() {
  50. typeWriter(element, text, i + 1);
  51. }, 50); // 调整打字速度
  52. } else {
  53. document.getElementById('chat-output').appendChild(element);
  54. }
  55. }
  56. </script>
  57. </body>
  58. </html>

总结

本文介绍了如何使用 Spring Boot 和 WebSocket 实现类似 ChatGPT 的流式回复效果,并详细讲解了每个步骤的技术原理和代码实现。通过这种方式,我们可以构建更加 engaging 和 interactive 的 Web 应用,为用户带来全新的体验。

声明:本文内容由网友自发贡献,转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号