Java Server-Sent Events没有数据

问题:做可视化大屏,开始有数据,刷新几次页面之后就没数据了很奇怪,前端的请求在不停的重新连接。

解决方案:因为之前的代码中使用了单个 SseEmitter 实例,这导致旧的连接没有正确关闭而产生冲突。可以通过为每个请求创建一个新的 SseEmitter 实例来优化代码,同时确保旧的实例被适当地清理。

示例:

import com.alibaba.fastjson.JSONObject;
import org.checkerframework.checker.units.qual.A;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.MediaType;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
import xin.admin.domain.sse.NotificationSSE;

import javax.servlet.http.HttpServletRequest;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

@RestController
@RequestMapping("/admin/homePage")
public class NotificationSSEController {

    // 线程池,让sse异步操作
    private final ExecutorService executorService = Executors.newCachedThreadPool();

    // 使用Map来存储每个客户端的SseEmitter
    private final Map<String, SseEmitter> emitters = new ConcurrentHashMap<>();

    // 要发送的数据
    public static NotificationSSE data = new NotificationSSE();

    @CrossOrigin
    @RequestMapping(value = "/notification", produces = MediaType.TEXT_EVENT_STREAM_VALUE)
    public SseEmitter stream(HttpServletRequest request) {
        String clientId = getClientId(request);
        SseEmitter emitter = new SseEmitter(24L * 60 * 60 * 1000);

        emitters.put(clientId, emitter);

        emitter.onCompletion(() -> emitters.remove(clientId));
        emitter.onTimeout(() -> emitters.remove(clientId));
        emitter.onError((e) -> emitters.remove(clientId));

        sendNotification(); // 当客户端连接时立即发送通知
        return emitter;
    }

    @Scheduled(fixedRate = 1000 * 60 * 10)
    public void heartbeat() {
        sendNotification();
    }

    // B函数:负责SSE发送
    public void sendNotification() {
        emitters.forEach((clientId, emitter) -> {
            executorService.execute(() -> {
                try {
                    emitter.send(SseEmitter.event()
                            .id(String.valueOf(System.currentTimeMillis()))
                            .data(JSONObject.toJSONString(data)));
                } catch (Exception e) {
                    emitter.completeWithError(e);
                }
            });
        });
    }

    // 生成或获取客户端ID
    private String getClientId(HttpServletRequest request) {
        // 这里可以根据实际情况生成唯一的客户端ID
        return request.getSession(true).getId();
    }
}

  • 9
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中实现server-sent events,可以使用Servlet 3.0规范中的异步支持。具体步骤如下: 1. 在Servlet中启用异步支持 ```java @WebServlet(urlPatterns = "/sse", asyncSupported = true) public class SseServlet extends HttpServlet { // ... } ``` 2. 获取AsyncContext对象 ```java protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { AsyncContext asyncContext = request.startAsync(); // ... } ``` 3. 设置响应的Content-Type为text/event-stream ```java response.setContentType("text/event-stream"); ``` 4. 通过AsyncContext对象获取响应的输出流,实现数据的推送 ```java PrintWriter out = response.getWriter(); out.write("data: Hello\n\n"); out.flush(); ``` 5. 通过AsyncContext对象设置超时时间和完成回调函数 ```java asyncContext.setTimeout(0); // 禁用超时 asyncContext.addListener(new AsyncListener() { @Override public void onComplete(AsyncEvent event) throws IOException { // 关闭输出流等资源 } // 其他回调函数 }); ``` 完整示例代码如下: ```java @WebServlet(urlPatterns = "/sse", asyncSupported = true) public class SseServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { AsyncContext asyncContext = request.startAsync(); response.setContentType("text/event-stream"); PrintWriter out = response.getWriter(); asyncContext.setTimeout(0); // 禁用超时 asyncContext.addListener(new AsyncListener() { @Override public void onComplete(AsyncEvent event) throws IOException { out.close(); } @Override public void onTimeout(AsyncEvent event) throws IOException { out.close(); } // 其他回调函数 }); while (true) { out.write("data: Hello\n\n"); out.flush(); try { Thread.sleep(1000); } catch (InterruptedException e) { break; } } } } ``` 上述代码中的while循环可以替换成其他的数据推送逻辑。需要注意的是,由于server-sent events协议是基于HTTP的长连接,因此需要在客户端实现相应的事件监听和重连机制。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值