Heim >Java >javaLernprogramm >Wie Spring Boot SSE verwendet, um Daten an das Frontend zu übertragen

Wie Spring Boot SSE verwendet, um Daten an das Frontend zu übertragen

WBOY
WBOYnach vorne
2023-05-10 17:31:062926Durchsuche

Vorwort

SSE ist einfach eine Technologie, bei der der Server Daten aktiv an das Front-End sendet. Es ist unidirektional, was bedeutet, dass das Front-End keine Daten an den Server senden kann. SSE eignet sich für Nachrichten-Push, Überwachung und andere Szenarien, die nur Server-Push-Daten erfordern. Das Folgende ist eine einfache Simulation mit Spring Boot, um Fortschrittsdaten an das Front-End zu übertragen.

Serverseite

Bei der Verwendung in Spring Boot ist es am besten, die von Spring Web bereitgestellte SseEmitter-Klasse zu verwenden. Zu Beginn habe ich die im Internet erwähnte Methode verwendet, um den Inhalt festzulegen. Es wurde festgestellt, dass das Frontend die Verbindung jedes Mal neu erstellte. Lesen Sie abschließend diesen Artikel, um den endgültigen gewünschten Effekt zu erzielen:

SSE-Tool-Klasse

SSEServer.java

package vip.huhailong.catchat.sse;

import lombok.extern.slf4j.Slf4j;
import org.springframework.http.MediaType;
import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Consumer;

/**
 * @author Huhailong
 */
@Slf4j
public class SSEServer {

    /**
     * 当前连接数
     */
    private static AtomicInteger count = new AtomicInteger(0);

    private static Map<String, SseEmitter> sseEmitterMap = new ConcurrentHashMap<>();

    public static SseEmitter connect(String userId){
        //设置超时时间,0表示不过期,默认是30秒,超过时间未完成会抛出异常
        SseEmitter sseEmitter = new SseEmitter(0L);
        //注册回调
        sseEmitter.onCompletion(completionCallBack(userId));
        sseEmitter.onError(errorCallBack(userId));
        sseEmitter.onTimeout(timeOutCallBack(userId));
        sseEmitterMap.put(userId,sseEmitter);
        //数量+1
        count.getAndIncrement();
        log.info("create new sse connect ,current user:{}",userId);
        return sseEmitter;
    }
    /**
     * 给指定用户发消息
     */
    public static void sendMessage(String userId, String message){
        if(sseEmitterMap.containsKey(userId)){
            try{
                sseEmitterMap.get(userId).send(message);
            }catch (IOException e){
                log.error("user id:{}, send message error:{}",userId,e.getMessage());
                e.printStackTrace();
            }
        }
    }

    /**
     * 想多人发送消息,组播
     */
    public static void groupSendMessage(String groupId, String message){
        if(sseEmitterMap!=null&&!sseEmitterMap.isEmpty()){
            sseEmitterMap.forEach((k,v) -> {
                try{
                    if(k.startsWith(groupId)){
                        v.send(message, MediaType.APPLICATION_JSON);
                    }
                }catch (IOException e){
                    log.error("user id:{}, send message error:{}",groupId,message);
                    removeUser(k);
                }
            });
        }
    }
    public static void batchSendMessage(String message) {
        sseEmitterMap.forEach((k,v)->{
            try{
                v.send(message,MediaType.APPLICATION_JSON);
            }catch (IOException e){
                log.error("user id:{}, send message error:{}",k,e.getMessage());
                removeUser(k);
            }
        });
    }
    /**
     * 群发消息
     */
    public static void batchSendMessage(String message, Set<String> userIds){
        userIds.forEach(userId->sendMessage(userId,message));
    }
    public static void removeUser(String userId){
        sseEmitterMap.remove(userId);
        //数量-1
        count.getAndDecrement();
        log.info("remove user id:{}",userId);
    }
    public static List<String> getIds(){
        return new ArrayList<>(sseEmitterMap.keySet());
    }
    public static int getUserCount(){
        return count.intValue();
    }
    private static Runnable completionCallBack(String userId) {
        return () -> {
            log.info("结束连接,{}",userId);
            removeUser(userId);
        };
    }
    private static Runnable timeOutCallBack(String userId){
        return ()->{
            log.info("连接超时,{}",userId);
            removeUser(userId);
        };
    }
    private static Consumer<Throwable> errorCallBack(String userId){
        return throwable -> {
            log.error("连接异常,{}",userId);
            removeUser(userId);
        };
    }
}

Die obige Klasse kann als SSE-Tool-Klasse betrachtet werden. Lassen Sie uns sie unten verwenden

Erstellt die Controller-Ebene SSEController.java

package vip.huhailong.catchat.controller;

import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
import vip.huhailong.catchat.sse.SSEServer;

/**
 * @author Huhailong
 */
@Slf4j
@RestController
@CrossOrigin
@RequestMapping("/sse")
public class SSEController {

    @GetMapping("/connect/{userId}")
    public SseEmitter connect(@PathVariable String userId){
        return SSEServer.connect(userId);
    }

    @GetMapping("/process")
    public void sendMessage() throws InterruptedException {
        for(int i=0; i<=100; i++){
            if(i>50&&i<70){
                Thread.sleep(500L);
            }else{
                Thread.sleep(100L);
            }
            SSEServer.batchSendMessage(String.valueOf(i));
        }
    }
}

Die obige Verbindung wird zum Herstellen einer Verbindung mit sse verwendet. Zu diesem Zeitpunkt wurde die Verbindung erstellt und dann wird die folgende Prozessschnittstelle zum Übertragen von Daten verwendet Um den Effekt eines Fortschrittsbalkens zu erzielen, wird eine Zahl gedrückt. Um den Effekt deutlich zu machen, verlangsame ich mich beim Drücken auf 50 bis 70, und der Rest beträgt 100 ms

Front-End-Code

rrree

Endeffekt:

Wie Spring Boot SSE verwendet, um Daten an das Frontend zu übertragen

Das obige ist der detaillierte Inhalt vonWie Spring Boot SSE verwendet, um Daten an das Frontend zu übertragen. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:yisu.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen