springboot -sse -flux 服務(wù)器推送消息的方法
先說(shuō)BUG處理,遇到提示異步問(wèn)題 Async support must be enabled on a servlet and for all filters involved in async request processing. This is done in Java code using the Servlet API or by adding "<async-supported>true</async-supported>" to servlet and filter declarations in web.xml.
springboot在@WebFilter注解處,加入urlPatterns = { "/*" },asyncSupported = true
springmvc在web.xml處理
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee"
xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd"
version="3.0">
<filter-mapping>
<filter-name>shiroFilter</filter-name>
<url-pattern>/*</url-pattern>
<dispatcher>REQUEST</dispatcher>
<dispatcher>ASYNC</dispatcher>
</filter-mapping>demo1,服務(wù)器間隔一定時(shí)間推送內(nèi)容
接口方法
@GetMapping(path = "/sse/{userId}",produces = MediaType.TEXT_EVENT_STREAM_VALUE )
public Flux<ServerSentEvent<String>> sse(@PathVariable String userId) {
// 每?jī)擅胪扑鸵淮?
return Flux.interval(Duration.ofSeconds(2)).map(seq->
Tuples.of(seq, LocalDateTime.now())).log()//序號(hào)和時(shí)間
.map(data-> ServerSentEvent.<String>builder().id(userId).data(data.getT1().toString()).build());//推送內(nèi)容
}2.前端代碼
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
<meta charset="UTF-8"/>
<title>服務(wù)器推送事件</title>
</head>
<body>
<div>
<div id="data"></div>
<div id="result"></div><br/>
</div>
<script th:inline="javascript" >
//服務(wù)器推送事件
if (typeof (EventSource) !== "undefined") {
var source1 = new EventSource("http://localhost:9000/api/admin/test/sse/1");
//當(dāng)抓取到消息時(shí)
source1.onmessage = function (evt) {
document.getElementById("data").innerHTML = document.getElementById("data").innerHTML+"股票行情:" + evt.data;
};
} else {
//注意:ie瀏覽器不支持
document.getElementById("result").innerHTML = "抱歉,你的瀏覽器不支持 server-sent 事件...";
var xhr;
var xhr2;
if (window.XMLHttpRequest){
//IE7+, Firefox, Chrome, Opera, Safari瀏覽器支持該方法
xhr=new XMLHttpRequest();
xhr2=new XMLHttpRequest();
}else{
//IE6, IE5 瀏覽器不支持,使用ActiveXObject方法代替
xhr=new ActiveXObject("Microsoft.XMLHTTP");
xhr2=new ActiveXObject("Microsoft.XMLHTTP");
}
console.log(xhr);
console.log(xhr2);
xhr.open('GET', '/sse/countDown');
xhr.send(null);//發(fā)送請(qǐng)求
xhr.onreadystatechange = function() {
console.log("s響應(yīng)狀態(tài):" + xhr.readyState);
//2是空響應(yīng),3是響應(yīng)一部分,4是響應(yīng)完成
if (xhr.readyState > 2) {
//這兒可以使用response(對(duì)應(yīng)json)與responseText(對(duì)應(yīng)text)
var newData = xhr.response.substr(xhr.seenBytes);
newData = newData.replace(/\n/g, "#");
newData = newData.substring(0, newData.length - 1);
var data = newData.split("#");
console.log("獲取到的數(shù)據(jù):" + data);
document.getElementById("result").innerHTML = data;
//長(zhǎng)度重新賦值,下次截取時(shí)需要使用
xhr.seenBytes = xhr.response.length;
}
}
xhr2.open('GET', '/sse/retrieve');
xhr2.send(null);//發(fā)送請(qǐng)求
xhr2.onreadystatechange = function() {
console.log("s響應(yīng)狀態(tài):" + xhr2.readyState);
//0: 請(qǐng)求未初始化,2 請(qǐng)求已接收,3 請(qǐng)求處理中,4 請(qǐng)求已完成,且響應(yīng)已就緒
if (xhr2.readyState > 2) {
//這兒可以使用response(對(duì)應(yīng)json)與responseText(對(duì)應(yīng)text)
var newData1 = xhr2.response.substr(xhr2.seenBytes);
newData1 = newData1.replace(/\n/g, "#");
newData1 = newData1.substring(0, newData1.length - 1);
var data1 = newData1.split("#");
console.log("獲取到的數(shù)據(jù):" + data1);
document.getElementById("data").innerHTML = data1;
//長(zhǎng)度重新賦值,下次截取時(shí)需要使用
xhr2.seenBytes = xhr2.response.length;
}
}
}
</script>
</body>
</html>demo2 訂閱服務(wù)器消息,服務(wù)器send推送消息完成后,關(guān)閉sse.close
1.接口方法以及工具類
@GetMapping(path = "/sse/sub",produces = MediaType.TEXT_EVENT_STREAM_VALUE )
public SseEmitter subscribe(@RequestParam String questionId,HttpServletResponse response) {
// 簡(jiǎn)單異步發(fā)消息 ====
//questionId 訂閱id,id對(duì)應(yīng)了sse對(duì)象
new Thread(() -> {
try {
Thread.sleep(1000);
for (int i = 0; i < 10; i++) {
Thread.sleep(500);
SSEUtils.pubMsg(questionId, questionId + " - kingtao come " + i);
}
} catch (Exception e) {
e.printStackTrace();
} finally {
// 消息發(fā)送完關(guān)閉訂閱
SSEUtils.closeSub(questionId);
}
}).start();
// =================
return SSEUtils.addSub(questionId);
}工具類
import org.springframework.web.servlet.mvc.method.annotation.SseEmitter;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
public class SSEUtils {
// timeout
private static Long DEFAULT_TIME_OUT = 2*60*1000L;
// 訂閱表
private static Map<String, SseEmitter> subscribeMap = new ConcurrentHashMap<>();
/** 添加訂閱 */
public static SseEmitter addSub(String questionId) {
if (null == questionId || "".equals(questionId)) {
return null;
}
SseEmitter emitter = subscribeMap.get(questionId);
if (null == emitter) {
emitter = new SseEmitter(DEFAULT_TIME_OUT);
subscribeMap.put(questionId, emitter);
}
return emitter;
}
/** 發(fā)消息 */
public static void pubMsg(String questionId, String msg) {
SseEmitter emitter = subscribeMap.get(questionId);
if (null != emitter) {
try {
// 更規(guī)范的消息結(jié)構(gòu)看源碼
emitter.send(SseEmitter.event().data(msg));
} catch (Exception e) {
// e.printStackTrace();
}
}
}
/**
* 關(guān)閉訂閱
* @param questionId
*/
public static void closeSub(String questionId) {
SseEmitter emitter = subscribeMap.get(questionId);
if (null != emitter) {
try {
emitter.complete();
subscribeMap.remove(questionId);
} catch (Exception e) {
e.printStackTrace();
}
}
}
}2.前端代碼
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>sse</title>
</head>
<body>
<div>
<label>問(wèn)題id</label>
<input type="text" id="questionId">
<button onclick="subscribe()">訂閱</button>
<hr>
<label>F12-console控制臺(tái)查看消息</label>
</div>
<script>
function subscribe() {
let questionId = document.getElementById('questionId').value;
let url = 'http://localhost:9000/api/admin/test/sse/sub?questionId=' + questionId;
let eventSource = new EventSource(url);
eventSource.onmessage = function (e) {
console.log(e.data);
};
eventSource.onopen = function (e) {
console.log(e,1);
// todo
};
eventSource.onerror = function (e) {
// todo
console.log(e,2);
eventSource.close()
};
}
</script>
</body>
</html>到此這篇關(guān)于springboot -sse -flux 服務(wù)器推送消息的文章就介紹到這了,更多相關(guān)springboot sse flux 服務(wù)器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
PowerJob分布式任務(wù)調(diào)度源碼流程解讀
這篇文章主要為大家介紹了PowerJob分布式任務(wù)調(diào)度源碼流程解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-02-02
Spring Cloud Zuul路由網(wǎng)關(guān)服務(wù)過(guò)濾實(shí)現(xiàn)代碼
這篇文章主要介紹了Spring Cloud Zuul路由網(wǎng)關(guān)服務(wù)過(guò)濾實(shí)現(xiàn)代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-04-04
Java模擬新浪和騰訊自動(dòng)登錄并發(fā)送微博
這篇文章主要為大家詳細(xì)介紹了Java模擬新浪和騰訊自動(dòng)登錄并發(fā)送微博功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-07-07
簡(jiǎn)單了解Java中多態(tài)的基礎(chǔ)知識(shí)
這篇文章主要介紹了簡(jiǎn)單了解Java中的多態(tài),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-09-09
解析Java和Eclipse中加載本地庫(kù)(.dll文件)的詳細(xì)說(shuō)明
本篇文章是對(duì)Java和Eclipse中加載本地庫(kù)(.dll文件)進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
詳解Spring AOP 攔截器的基本實(shí)現(xiàn)
本篇文章主要介紹了詳解Spring AOP 攔截器的基本實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-03-03

