日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

springmvc(18)使用WebSocket 和 STOMP 实现消息功能

發布時間:2023/12/3 c/c++ 39 豆豆
生活随笔 收集整理的這篇文章主要介紹了 springmvc(18)使用WebSocket 和 STOMP 实现消息功能 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.
【0】README 1)本文旨在 介紹如何 利用 WebSocket 和 STOMP 實現消息功能; 2)要知道, WebSocket 是發送和接收消息的 底層API,而SockJS 是在 WebSocket 之上的 API;最后 STOMP(面向消息的簡單文本協議)是基于 SockJS 的高級API (干貨——簡而言之,WebSocket 是底層協議,SockJS 是WebSocket 的備選方案,也是 底層協議,而 STOMP 是基于 WebSocket(SockJS) 的上層協議) 3)broker==經紀人,代理; 4)當然,你可以直接跳轉到 STOMP 知識(章節【3】);

【1】WebSocket 1)intro:WebSocket 協議提供了 通過一個套接字實現全雙工通信的功能。也能夠實現 web ?瀏覽器 和 server 間的 異步通信, 全雙工意味著 server 與 瀏覽器間 可以發送和接收消息。
【1.1】使用 spring 的低層級 WebSocket API 1)intro:為了在 spring 中 使用較低層級的 API 來處理消息。有如下方案: scheme1)我們必須編寫一個實現 WebSocketHandler: public interface WebSocketHandler { void afterConnectionEstablished(WebSocketSession session) throws Exception; void handleMessage(WebSocketSession session, WebSocketMessage<?> message) throws Exception; void handleTransportError(WebSocketSession session,Throwable exception) throws Exception; void afterConnectionClosed(WebSocketSession session,CloseStatus closeStatus) throws Exception; boolean supportsPartialMessages(); } scheme2)當然,我們也可以擴展 AbstractWebSocketHandler(更加簡單一點); // you can also extends TextWebSocketHandler public class ChatTextHandler extends AbstractWebSocketHandler {// handle text msg.@Overrideprotected void handleTextMessage(WebSocketSession session,TextMessage message) throws Exception {session.sendMessage(new TextMessage("hello world."));} } 對以上代碼的分析(Analysis): 當然了,我們還可以重載其他三個方法:? ?? handleBinaryMessage() handlePongMessage() handleTextMessage() scheme3)也可以擴展 TextWebSocketHandler(文本 WebSocket 處理器), 不在擴展AbstractWebSocketHandler?,?TextWebSocketHandler 繼承?AbstractWebSocketHandler?;
2)你可能會關系建立和關閉連接感興趣??梢灾剌d afterConnectionEstablished() and afterConnectionClosed(): // 當新連接建立的時候,被調用; public void afterConnectionEstablished(WebSocketSession session) throws Exception { logger.info("Connection established"); } // 當連接關閉時被調用; @Override public void afterConnectionClosed( WebSocketSession session, CloseStatus status) throws Exception { logger.info("Connection closed. Status: " + status); } 3)現在已經有了 message handler 類了,下面對其進行配置,配置到 springmvc 的運行環境中。 @Configuration @EnableWebSocket public class WebSocketConfig implements WebSocketConfigurer{@Overridepublic void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {registry.addHandler(getTextHandler(), "/websocket/p2ptext");} // 將 ChatTextHandler 處理器 映射到 /websocket/p2ptext 路徑下.@Beanpublic ChatTextHandler getTextHandler() {return new ChatTextHandler();} } 對上述代碼的分析(Analysis):registerWebSocketHandlers方法 是注冊消息處理器的關鍵: 通過 調用?WebSocketHandlerRegistry?.addHandler() 方法 來注冊信息處理器; Attention)server 端的 WebSocket 配置完畢,下面配置客戶端;
4)WebSocket 客戶端配置 4.1)client 發送 一個文本到 server,他監聽來自 server 的文本消息。下面代碼 展示了 利用 js 開啟一個原始的 WebSocket 并使用它來發送消息給server; 4.2)代碼如下: <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%><html lang="zh-CN"> <head> <meta charset="utf-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1"> <!-- 上述3個meta標簽*必須*放在最前面,任何其他內容都*必須*跟隨其后! --> <title>web socket</title><link href="<c:url value="/"/>bootstrap/css/bootstrap.min.css" rel="stylesheet"><!-- jQuery (necessary for Bootstrap's JavaScript plugins) --> <script src="<c:url value="/"/>bootstrap/jquery/jquery.min.js"></script> <!-- Include all compiled plugins (below), or include individual files as needed --> <script src="<c:url value="/"/>bootstrap/js/bootstrap.min.js"></script><script type="text/javascript"> $(document).ready(function() { websocket_client(); });function websocket_client() { var hostaddr = window.location.host + "<c:url value='/websocket/p2ptext' />"; var url = 'ws://' + hostaddr; var sock = new WebSocket(url);// 以下的 open(), onmessage(), onclose() // 對應到 ChatTextHandler 的 // afterConnectionEstablished(), handleTextMessage(), afterConnectionClosed();sock.open = function() { alert("open successfully."); sayMarco(); };sock.onmessage = function(e) { alert("onmessage"); alert(e); };sock.onclose = function() { alert("close"); };function sayMarco() { sock.send("this is the websocket client."); } } </script> </head><body> <div id="websocket"> websocket div. </div> </body> </html> error)這樣配置后, WebSocket 無法正常運行;
【2】應對不支持 WebSocket 的場景(引入 SockJS) 1)problem+solutions: 1.1)problem:許多瀏覽器不支持 WebSocket 協議; 1.2)solutions:?SockJS 是 WebSocket 技術的一種模擬。SockJS 會 盡可能對應 WebSocket API,但如果 WebSocket 技術 不可用的話,就會選擇另外的 通信方式協議;
2)SockJS 會優先選擇 WebSocket 協議,但是如果 WebSocket協議不可用的話,他就會從如下 方案中挑選最優可行方案: XHR streaming XDR streaming iFrame event source iFrame HTML file XHR polling XDR polling iFrame XHR polling JSONP polling 3)如何在 server 端配置 SockJS :添加?withSockJS() 方法; // 將 ChatTextHandler 映射到 /chat/text 路徑下.@Overridepublic void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {registry.addHandler(getTextHandler(), "/websocket/p2ptext").withSockJS(); // withSockJS() 方法聲明我們想要使用 SockJS 功能,如果WebSocket不可用的話,會使用 SockJS;}4)客戶端配置 SockJS, 想要確保 加載了 SockJS 客戶端; 4.1)具體做法是 依賴于 JavaScript 模塊加載器(如 require.js or curl.js) 還是簡單使用 <script> 標簽加載 JavaScript 庫。最簡單的方法是 使用 <script> 標簽從 SockJS CDN 中進行加載,如下所示: <script src="http://cdn.sockjs.org/sockjs-0.3.min.js"></script> Attention)用 WebJars 解析 Web資源(可選,有興趣的童鞋可以嘗試下) A1)在springmvc 配置中搭建一個 資源處理器,讓它負責解析路徑以 "webjars/**" 開頭的請求,這也是 WebJars 的標準路徑: @Override public void addResourceHandlers(ResourceHandlerRegistry registry) { registry.addResourceHandler("/webjars/**") .addResourceLocations("classpath:/META-INF/resources/webjars/"); } A2)在這個資源處理器 準備就緒后,我們可以在 web 頁面中使用 如下的 <script> 標簽加載 SockJS 庫; <script src="sockjs.min.js}"> </script> 5)處理加載 SockJS 客戶端庫以外,還要修改 兩行代碼: var url = 'p2ptext'; var sock = new SockJS(url); 對以上代碼的分析(Analysis):? A1)SockJS 所處理的URL 是 "http://" 或 "https://" 模式,而不是 "ws://" or ?"wss://" ; A2)其他的函數如 onopen, onmessage, and ?onclose ,SockJS 客戶端與 WebSocket 一樣;
6)SockJS 為 WebSocket 提供了 備選方案。但無論哪種場景,對于實際應用來說,這種通信形式層級過低。下面看一下如何 在 WebSocket 之上使用 STOMP協議,來為瀏覽器 和 server間的 通信增加適當的消息語義;(干貨——引入 STOMP—— Simple Text Oriented Message Protocol——面向消息的簡單文本協議)
【3】使用 STOMP消息 1)intro: 如何理解 STOMP 與 WebSocket 的關系: 1.1)假設 HTTP 協議 并不存在,只能使用 TCP 套接字來 編寫 web 應用,你可能認為這是一件瘋狂的 事情; 1.2)不過 幸好,我們有 HTTP協議,它解決了 web 瀏覽器發起請求以及 web 服務器響應請求的細節; 1.3)直接使用 WebSocket(SockJS) 就很類似于 使用 TCP 套接字來編寫 web 應用;因為沒有高層協議,因此就需要我們定義應用間所發送消息的語義,還需要確保 連接的兩端都能遵循這些語義; 1.4)同 HTTP 在 TCP 套接字上添加 請求-響應 模型層一樣,STOMP 在 WebSocket 之上提供了一個基于 幀的線路格式層,用來定義消息語義;(干貨——STOMP 在 WebSocket 之上提供了一個基于 幀的線路格式層,用來定義消息語義) 2)STOMP 幀:該幀由命令,一個或多個 頭信息 以及 負載所組成。如下就是發送 數據的一個 STOMP幀:(干貨——引入了 STOMP幀格式) SEND destination:/app/marco content-length:20{\"message\":\"Marco!\"} 對以上代碼的分析(Analysis):
A1)SEND:STOMP命令,表明會發送一些內容; A2)destination:頭信息,用來表示消息發送到哪里; A3)content-length:頭信息,用來表示 負載內容的 大小; A4)空行: A5)幀內容(負載)內容
3)STOMP幀 信息 最有意思的是 destination頭信息了:?它表明 STOMP 是一個消息協議,類似于 JMS 或 AMQP。消息會發送到 某個 目的地,這個 目的地實際上可能真的 有消息代理作為 支撐。另一方面,消息處理器 也可以監聽這些目的地,接收所發送過來的消息;
【3.1】啟用STOMP 消息功能 1)intro:spring 的消息功能是基于消息代理構建的,因此我們必須要配置一個 消息代理 和 其他的一些消息目的地;(干貨——spring 的消息功能是基于消息代理構建的) 2)如下代碼展現了 如何通過 java配置 啟用基于代理的的web 消息功能; (干貨——@EnableWebSocketMessageBroker 注解的作用: 能夠在 WebSocket 上啟用 STOMP)
package com.spring.spittr.web;import org.springframework.context.annotation.Configuration; import org.springframework.messaging.handler.annotation.MessageMapping; import org.springframework.messaging.simp.config.MessageBrokerRegistry; import org.springframework.web.socket.config.annotation.AbstractWebSocketMessageBrokerConfigurer; import org.springframework.web.socket.config.annotation.EnableWebSocketMessageBroker; import org.springframework.web.socket.config.annotation.StompEndpointRegistry;@Configuration @EnableWebSocketMessageBroker public class WebSocketConfig extends AbstractWebSocketMessageBrokerConfigurer {@Overridepublic void configureMessageBroker(MessageBrokerRegistry config) {config.enableSimpleBroker("/topic", "/queue");config.setApplicationDestinationPrefixes("/app");// 應用程序以 /app 為前綴,而 代理目的地以 /topic 為前綴.// js.url = "/spring13/app/hello" -> @MessageMapping("/hello") 注釋的方法.}@Overridepublic void registerStompEndpoints(StompEndpointRegistry registry) {registry.addEndpoint("/hello").withSockJS();// 在網頁上我們就可以通過這個鏈接 /server/hello ==<c:url value='/hello'></span> 來和服務器的WebSocket連接} } 對以上代碼的分析(Analysis): A1)EnableWebSocketMessageBroker注解表明: 這個配置類不僅配置了 WebSocket,還配置了 基于代理的 STOMP 消息; A2)它重載了?registerStompEndpoints() 方法:將 "/hello" 路徑 注冊為 STOMP 端點。這個路徑與之前發送和接收消息的目的路徑有所不同, 這是一個端點,客戶端在訂閱或發布消息 到目的地址前,要連接該端點,即 用戶發送請求 url='/server/hello' 與 STOMP server 進行連接,之后再轉發到 訂閱url;(server== name of your springmvc project (干貨——端點的作用——客戶端在訂閱或發布消息 到目的地址前,要連接該端點) A3)它重載了 configureMessageBroker() 方法:配置了一個 簡單的消息代理。如果不重載,默認case下,會自動配置一個簡單的 內存消息代理,用來處理 "/topic" 為前綴的消息。但經過重載后,消息代理將會處理前綴為 "/topic" and "/queue" 消息。 A4)之外:發送應用程序的消息將會帶有 "/app" 前綴,下圖展現了 這個配置中的 消息流;
對上述處理step的分析(Analysis): A1)應用程序的目的地 以 "/app" 為前綴,而代理的目的地以 "/topic" 和 "/queue" 作為前綴; A2)以應用程序為目的地的消息將會直接路由到 帶有 @MessageMapping 注解的控制器方法中;(干貨——?@MessageMapping的作用) A3)而發送到 代理上的消息,包括 @MessageMapping注解方法的返回值所形成的消息,將會路由到 代理上,并最終發送到 訂閱這些目的地客戶端; (干貨——client 連接地址和 發送地址是不同的,以本例為例,前者是/server/hello, 后者是/server/app/XX,先連接后發送)
【3.1.1】啟用 STOMP 代理中繼 1)intro:在生成環境下,可能會希望使用 真正支持 STOMP 的代理來支持 WebSocket 消息,如RabbitMQ 或 ActiveMQ。這樣的代理提供了可擴展性和健壯性更好的消息功能,當然,他們也支持 STOMP 命令; 2)如何 使用 STOMP 代理來替換內存代理,代碼如下: @Overridepublic void configureMessageBroker(MessageBrokerRegistry registry) {// 啟用了 STOMP 代理中繼功能,并將其代理目的地前綴設置為 /topic and /queue .registry.enableStompBrokerRelay("/queue", "/topic").setRelayPort(62623);registry.setApplicationDestinationPrefixes("/app"); // 應用程序目的地.} 對以上代碼的分析(Analysis):(干貨——STOMP代理前綴和 應用程序前綴的意義) A1)方法第一行啟用了 STOMP 代理中繼功能: 并將其目的地前綴設置為 "/topic" or "/queue" ;spring就能知道 所有目的地前綴為 "/topic" or "/queue" 的消息都會發送到 STOMP 代理中; A2)方法第二行設置了 應用的前綴為 "app":所有目的地以 "/app" 打頭的消息(發送消息url not 連接url)都會路由到 帶有 @MessageMapping 注解的方法中,而不會發布到 代理隊列或主題中; 3)下圖闡述了 代理中繼如何 應用于 spring 的 STOMP 消息處理之中。與 上圖的 關鍵區別在于: 這里不再模擬STOMP 代理的功能,而是由 代理中繼將消息傳送到一個 真正的消息代理來進行處理;
Attention) A1)enableStompBrokerRelay() and setApplicationDestinationPrefixes() 方法都可以接收變長 參數; A2)默認情況下: STOMP 代理中繼會假設 代理監聽 localhost 的61613 端口,并且 client 的 username 和password 均為 guest。當然你也可以自行定義; @Override public void configureMessageBroker(MessageBrokerRegistry registry) { registry.enableStompBrokerRelay("/topic", "/queue") .setRelayHost("rabbit.someotherserver") .setRelayPort(62623) .setClientLogin("marcopolo") .setClientPasscode("letmein01"); registry.setApplicationDestinationPrefixes("/app", "/foo"); } // setXXX()方法 是可選的 【3.2】 處理來自客戶端的 STOMP 消息 1)借助 @MessageMapping 注解能夠 在 控制器中處理 STOMP 消息
package com.spring.spittr.web;import org.springframework.messaging.handler.annotation.MessageMapping; import org.springframework.messaging.handler.annotation.SendTo; import org.springframework.stereotype.Controller;import com.spring.pojo.Greeting; import com.spring.pojo.HelloMessage;@Controller public class GreetingController {@MessageMapping("/hello")@SendTo("/topic/greetings")public Greeting greeting(HelloMessage message) throws Exception {System.out.println("receiving " + message.getName());System.out.println("connecting successfully.");return new Greeting("Hello, " + message.getName() + "!");} }對以上代碼的分析(Analysis): A1)@MessageMapping注解:表示?handleShout()方法能夠處理 指定目的地上到達的消息; A2)這個目的地(消息發送目的地url)就是 "/server/app/hello",其中 "/app" 是 隱含的 ,"/server" 是 springmvc 項目名稱;
2)因為我們現在處理的 不是 HTTP,所以無法使用 spring 的 HttpMessageConverter 實現 將負載轉換為Shout 對象。Spring 4.0 提供了幾個消息轉換器如下:(Attention, 如果是傳輸json數據的話,定要添加 Jackson jar 包到你的springmvc 項目中,不然連接不會成功的)

【3.2.1】處理訂閱(@SubscribeMapping注解) 1)@SubscribeMapping注解 的方法:當收到 STOMP 訂閱消息的時候,帶有 @SubscribeMapping 注解 的方法將會觸發;其也是通過 AnnotationMethodMessageHandler 來接收消息的; 2)@SubscribeMapping注解的應用場景:實現 請求-回應模式。在請求-回應模式中,客戶端訂閱一個目的地,然后預期在這個目的地上 獲得一個一次性的 響應;(干貨——引入了@SubsribeMapping注解實現 請求-回應模式) 2.1)看個荔枝: @SubscribeMapping({"/marco"}) public Shout handleSubscription() { Shout outgoing = new Shout(); outgoing.setMessage("Polo!"); return outgoing; } 對以上代碼的分析(Analysis): A1)@SubscribeMapping注解 的方法來處理 對 "/app/macro" 目的地訂閱(與 @MessageMapping類似,"/app" 是隱含的 ); A2)請求-回應模式與 HTTP GET 的全球-響應模式差不多: 關鍵區別在于, HTTP GET 請求是同步的,而訂閱的全球-回應模式是異步的,這樣客戶端能夠在回應可用時再去處理,而不必等待;(干貨——HTTP GET 請求是同步的,而訂閱的請求-回應模式是異步的)
【3.2.2】編寫 JavaScript 客戶端 1)intro:借助 STOMP 庫,通過 JavaScript發送消息
<script type="text/javascript">var stompClient = null;function setConnected(connected) {document.getElementById('connect').disabled = connected;document.getElementById('disconnect').disabled = !connected;document.getElementById('conversationDiv').style.visibility = connected ? 'visible' : 'hidden';document.getElementById('response').innerHTML = '';}function connect() {var socket = new SockJS("<c:url value='/hello'/>");stompClient = Stomp.over(socket);stompClient.connect({}, function(frame) {setConnected(true);console.log('Connected: ' + frame);stompClient.subscribe('/topic/greetings', function(greeting){showGreeting(JSON.parse(greeting.body).content);});});}function disconnect() {if (stompClient != null) {stompClient.disconnect();}setConnected(false);console.log("Disconnected");}function sendName() {var name = document.getElementById('name').value;stompClient.send("/app/hello", {}, JSON.stringify({ 'name': name }));}function showGreeting(message) {var response = document.getElementById('response');var p = document.createElement('p');p.style.wordWrap = 'break-word';p.appendChild(document.createTextNode(message));response.appendChild(p);}</script> 對以上代碼的 分析(Analysis):?以上代碼連接“/hello” 端點并發送 ”name“; 2)stompClient.send("/app/hello", {}, JSON.stringify({'name':name})): 第一個參數:json 負載消息發送的 目的地; 第二個參數:是一個頭信息的Map,它會包含在 STOMP 幀中;第三個參數:負載消息; (干貨—— stomp client 連接地址 和 發送地址不一樣的,連接地址為 <c:url value='/hello'/> ==localhost:8080/springmvc_project_name/hello , 而 發送地址為 '/app/hello',這里要當心)

downloading these files below from?https://github.com/pacosonTang/SpringInAction/tree/master/spring18 <script src="<c:url value="/resources/sockjs-1.1.1.js" />"></script><script src="<c:url value="/resources/stomp.js" />"></script> //this line.function connect() {var socket = new SockJS("<c:url value='/hello'/>");stompClient = Stomp.over(socket);stompClient.connect({}, function(frame) {setConnected(true);console.log('Connected: ' + frame);stompClient.subscribe('/topic/greetings', function(greeting){showGreeting(JSON.parse(greeting.body).content);});stompClient.subscribe('/app/macro',function(greeting){alert(JSON.parse(greeting.body).content);showGreeting(JSON.parse(greeting.body).content);});});}function sendName() {var name = document.getElementById('name').value;stompClient.send("/app/hello", {}, JSON.stringify({ 'name': name }));}package com.spring.spittr.web;import org.springframework.beans.factory.annotation.Autowired; import org.springframework.messaging.handler.annotation.MessageMapping; import org.springframework.messaging.handler.annotation.SendTo; import org.springframework.messaging.simp.SimpMessageSendingOperations; import org.springframework.messaging.simp.SimpMessagingTemplate; import org.springframework.messaging.simp.annotation.SubscribeMapping; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import org.springframework.web.bind.annotation.RequestParam;import com.spring.pojo.Greeting; import com.spring.pojo.HelloMessage;@Controller public class GreetingController {// @MessageMapping defines the sending addr for client.// 消息發送地址: /server/app/hello@MessageMapping("/hello")@SendTo("/topic/greetings")public Greeting greeting(HelloMessage message) throws Exception {System.out.println("receiving " + message.getName());System.out.println("connecting successfully.");return new Greeting("Hello, " + message.getName() + "!");}@SubscribeMapping("/macro")public Greeting handleSubscription() {System.out.println("this is the @SubscribeMapping('/marco')");Greeting greeting = new Greeting("i am a msg from SubscribeMapping('/macro').");return greeting;}/*@MessageMapping("/feed")@SendTo("/topic/feed")public Greeting greetingForFeed(HelloMessage message) throws Exception {System.out.println("receiving " + message.getName());System.out.println("connecting successfully.");return new Greeting("i am /topic/feed, hello " + message.getName() + "!");}*/// private SimpMessagingTemplate template;// SimpMessagingTemplate implements SimpMessageSendingOperations. private SimpMessageSendingOperations template;@Autowiredpublic GreetingController(SimpMessageSendingOperations template) {this.template = template;}@RequestMapping(path="/feed", method=RequestMethod.POST)public void greet(@RequestParam String greeting) {String text = "you said just now " + greeting;this.template.convertAndSend("/topic/feed", text);} }package com.spring.spittr.web;import org.springframework.context.annotation.Configuration; import org.springframework.messaging.handler.annotation.MessageMapping; import org.springframework.messaging.simp.config.MessageBrokerRegistry; import org.springframework.web.socket.config.annotation.AbstractWebSocketMessageBrokerConfigurer; import org.springframework.web.socket.config.annotation.EnableWebSocketMessageBroker; import org.springframework.web.socket.config.annotation.StompEndpointRegistry;@Configuration @EnableWebSocketMessageBroker public class WebSocketConfig extends AbstractWebSocketMessageBrokerConfigurer {@Overridepublic void configureMessageBroker(MessageBrokerRegistry config) {config.enableSimpleBroker("/topic", "/queue");config.setApplicationDestinationPrefixes("/app");// 應用程序以 /app 為前綴,而 代理目的地以 /topic 為前綴.// js.url = "/spring13/app/hello" -> @MessageMapping("/hello") 注釋的方法.}@Overridepublic void registerStompEndpoints(StompEndpointRegistry registry) {registry.addEndpoint("/hello").withSockJS();// 在網頁上我們就可以通過這個鏈接 /server/hello 來和服務器的WebSocket連接} } package com.spring.spittr.web;import java.io.IOException;import org.springframework.context.MessageSource; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.Import; import org.springframework.context.support.ResourceBundleMessageSource; import org.springframework.core.io.FileSystemResource; import org.springframework.web.multipart.MultipartResolver; import org.springframework.web.multipart.commons.CommonsMultipartResolver; import org.springframework.web.servlet.ViewResolver; import org.springframework.web.servlet.config.annotation.DefaultServletHandlerConfigurer; import org.springframework.web.servlet.config.annotation.EnableWebMvc; import org.springframework.web.servlet.config.annotation.ViewControllerRegistry; import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter; import org.springframework.web.servlet.view.InternalResourceViewResolver; import org.springframework.web.servlet.view.tiles3.TilesConfigurer; import org.springframework.web.servlet.view.tiles3.TilesViewResolver;@Configuration @ComponentScan(basePackages = { "com.spring.spittr.web" }) @EnableWebMvc @Import({WebSocketConfig.class}) public class WebConfig extends WebMvcConfigurerAdapter {@Beanpublic TilesConfigurer tilesConfigurer() {TilesConfigurer tiles = new TilesConfigurer();tiles.setDefinitions(new String[] { "/WEB-INF/layout/tiles.xml" });tiles.setCheckRefresh(true);return tiles;}// config processing for static resources.@Overridepublic void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {configurer.enable();}// InternalResourceViewResolver @Beanpublic ViewResolver viewResolver1() {TilesViewResolver resolver = new TilesViewResolver();return resolver;}@Beanpublic ViewResolver viewResolver2() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/WEB-INF/views/");resolver.setSuffix(".jsp");resolver.setExposeContextBeansAsAttributes(true);resolver.setViewClass(org.springframework.web.servlet.view.JstlView.class);return resolver;}@Beanpublic MessageSource messageSource() {ResourceBundleMessageSource messageSource = new ResourceBundleMessageSource();messageSource.setBasename("messages"); return messageSource;}@Beanpublic MultipartResolver multipartResolver() throws IOException {CommonsMultipartResolver multipartResolver = new CommonsMultipartResolver();multipartResolver.setUploadTempDir(new FileSystemResource("/WEB-INF/tmp/spittr/uploads"));multipartResolver.setMaxUploadSize(2097152);multipartResolver.setMaxInMemorySize(0);return multipartResolver;} } 【3.3】發送消息到客戶端 1)intro:spring提供了兩種 發送數據到 client 的方法: method1)作為處理消息 或處理訂閱的附帶結果; method2)使用消息模板;
【3.3.1】在處理消息后,發送消息(server 對 client 請求的 響應消息) 1)intro:如果你想要在接收消息的時候,在響應中發送一條消息,修改方法簽名 不是void 類型即可, 如下:
@MessageMapping("/hello")@SendTo("/topic/greetings") //highlight line.public Greeting greeting(HelloMessage message) throws Exception {System.out.println("receiving " + message.getName());System.out.println("connecting successfully.");return new Greeting("Hello, " + message.getName() + "!");}對以上代碼的分析(Analysis):返回的對象將會進行轉換(通過消息轉換器) 并放到 STOMP 幀的負載中,然后發送給消息代理(消息代理分為 STOMP代理中繼 和 內存消息代理) 2)默認情況下:幀所發往的目的地會與 觸發 處理器方法的目的地相同。所以返回的對象 會寫入到 STOMP 幀的負載中,并發布到 "/topic/stomp" 目的地。不過,可以通過 @SendTo 注解,重載目的地;(干貨——注解?@SendTo 注解的作用)
代碼同上。
對以上代碼的分析(Analysis):
消息將會發布到 /topic/hello, 所有訂閱這個主題的應用都會收到這條消息;

3)@SubscriptionMapping 注解標注的方式也能發送一條消息,作為訂閱的回應。 3.1)看個荔枝: 通過為 控制器添加如下的方法,當客戶端訂閱的時候,將會發送一條 shout 信息: @SubscribeMapping("/macro") // defined in Controller. attention for addr '/macro' in server.public Greeting handleSubscription() {System.out.println("this is the @SubscribeMapping('/marco')");Greeting greeting = new Greeting("i am a msg from SubscribeMapping('/macro').");return greeting;} function connect() {var socket = new SockJS("<c:url value='/hello'/>");stompClient = Stomp.over(socket);stompClient.connect({}, function(frame) {setConnected(true);console.log('Connected: ' + frame);stompClient.subscribe('/topic/greetings', function(greeting){showGreeting(JSON.parse(greeting.body).content);});// starting line.stompClient.subscribe('/app/macro',function(greeting){alert(JSON.parse(greeting.body).content);showGreeting(JSON.parse(greeting.body).content);}); // ending line. attention for addr '/app/macro' in client.});} 對以上代碼的分析(Analysis):? A0)這個SubscribeMapping annotation標記的方法,是在訂閱的時候調用的,也就是說,基本是只執行一次的方法,client 調用定義在server 的 該 Annotation 標注的方法,它就會返回結果,不過經過代理。 A1)這里的?@SubscribeMapping 注解表明當 客戶端訂閱 "/app/macro" 主題的時候("/app"是應用目的地的前綴,注意,這里沒有加springmvc 項目名稱前綴), 將會調用?handleSubscription 方法。它所返回的shout 對象 將會進行轉換 并發送回client; A2)SubscribeMapping注解的區別在于: 這里的 Shout 消息將會直接發送給 client,不用經過 消息代理;但,如果為方法添加 @SendTo 注解的話,那么 消息將會發送到指定的目的地,這樣就會經過代理;(干貨——SubscribeMapping注解返回的消息直接發送到 client,不經過代理,而 @SendTo 注解的路徑,就會經過代理,然后再發送到 目的地)

【3.3.2】 在應用的任意地方發送消息 1)intro:spring 的 SimpMessagingTemplate 能夠在應用的任何地方發送消息,不必以接收一條消息為 前提; 2)看個荔枝: 讓首頁訂閱一個 STOMP主題,在 Spittle 創建的時候,該主題能夠收到 Spittle 更新時的 feed; 2.1)JavaScript 代碼: <script> var sock = new SockJS('spittr'); var stomp = Stomp.over(sock); stomp.connect('guest', 'guest', function(frame) { console.log('Connected'); stomp.subscribe("/topic/spittlefeed", handleSpittle); // highlight. }); function handleSpittle(incoming) { var spittle = JSON.parse(incoming.body); console.log('Received: ', spittle); var source = $("#spittle-template").html(); var template = Handlebars.compile(source); var spittleHtml = template(spittle); $('.spittleList').prepend(spittleHtml); } </script> 對以上代碼的分析(Analysis):?在連接到 STMOP 代理后,我們訂閱了 "/topic/spittlefeed" 主題,并指定當消息到達的是,由?handleSpittle()函數來處理 Spittle 更新。 2.2) server 端代碼:使用 SimpMessagingTemplate 將所有新創建的 Spittle 以消息的形式發布到 "/topic/feed" 主題上; @Service public class SpittleFeedServiceImpl implements SpittleFeedService { private SimpMessageSendingOperations messaging; @Autowired public SpittleFeedServiceImpl( SimpMessageSendingOperations messaging) { // 注入消息模板. this.messaging = messaging; } public void broadcastSpittle(Spittle spittle) { messaging.convertAndSend("/topic/spittlefeed", spittle); // 發送消息. } } 對以上代碼的分析(Analysis):? A1)配置 spring 支持 stomp 的一個附帶功能是 在spring應用上下文中已經包含了 Simple A2)在發布消息給 STOMP 主題的時候,所有訂閱該主題的客戶端都會收到消息。但有的時候,我們希望將消息發送給指定用戶;
【4】 為目標用戶發送消息 1)intro:在使用 srping 和 STOMP 消息功能的時候,有三種方式來利用認證用戶: way1)@MessageMapping and @SubscribeMapping 注解標注的方法 能夠使用 Principal 來獲取認證用戶; way2)@MessageMapping, @SubscribeMapping, and @MessageException 方法返回的值能夠以 消息的形式發送給 認證用戶; way3)SimpMessagingTemplate 能夠發送消息給特定用戶;
【4.1】在控制器中處理用戶的 消息 1)看個荔枝: 編寫一個控制器方法,根據傳入的消息創建新的Spittle 對象,并發送一個回應,表明 對象創建成功;(這種 REST也可以實現,不過它是同步的,而這里是異步的); 1.1)代碼如下:它會處理傳入的消息并將其存儲我 Spittle: @MessageMapping("/spittle") @SendToUser("/queue/notifications") public Notification handleSpittle( Principal principal, SpittleForm form) { Spittle spittle = new Spittle( principal.getName(), form.getText(), new Date()); spittleRepo.save(spittle); return new Notification("Saved Spittle"); } 1.2)該方法最后返回一個 新的 Notificatino,表明對象保存成功; 1.3)該方法使用了?@MessageMapping("/spittle") 注解,所以當有發往 "/app/spittle" 目的地的消息 到達時,該方法就會觸發;如果用戶已經認證的話,將會根據 STOMP 幀上的頭信息得到 Principal 對象; 1.4)@SendToUser注解: 指定了 Notification 要發送的 目的地?"/queue/notifications"; 1.5)表明上,?"/queue/notifications" 并不會與 特定用戶相關聯,但因為 這里使用的是?@SendToUser注解, 而不是?@SendTo,所以 就會發生更多的事情了; 2)看一下針對 控制器方法發布的 Notificatino 對象的目的地,客戶端該如何進行訂閱。 2.1)看個荔枝:考慮如下的 JavaScript代碼,它訂閱了一個 用戶特定的 目的地: stomp.subscribe("/user/queue/notifications", handleNotifications); 對以上代碼的分析(Analysis):這個目的地使用了 "/user" 作為前綴,在內部,以"/user" 為前綴的消息將會通過 UserDestinationMessageHandler 進行處理,而不是 AnnotationMethodMessageHandler 或 ?SimpleBrokerMessageHandler or StompBrokerRelayMessageHandler,如下圖所示:
Attention)UserDestinationMessageHandler?的主要任務: 是 將用戶消息重新路由到 某個用戶獨有的目的地上。 在處理訂閱的時候,它會將目標地址中的 "/user" 前綴去掉,并基于用戶 的會話添加一個后綴。如,對 ?"/user/queue/notifications" 的訂閱最后可能路由到 名為 "/queue/notifacations-user65a4sdfa" 目的地上;
【4.2】為指定用戶發送消息 1)intro:SimpMessagingTemplate還提供了 convertAndSendToUser() 方法,該方法能夠讓 我們給特定用戶發送消息; 2)我們在 web 應用上添加一個特性: 當其他用戶提交的 Spittle 提到某個用戶時,將會提醒該用戶(干貨——這難道不是 微博的 @ 功能嗎) 2.1)看個荔枝:如果Spittle 文本中包含 "@tangrong",那么我們就應該發送一條消息給 使用?tangrong 用戶名登錄的client,代碼實例如下: @Service public class SpittleFeedServiceImpl implements SpittleFeedService { private SimpMessagingTemplate messaging; // 實現用戶提及功能的正則表達式 private Pattern pattern = Pattern.compile("\\@(\\S+)"); @Autowired public SpittleFeedServiceImpl(SimpMessagingTemplate messaging) { this.messaging = messaging; } public void broadcastSpittle(Spittle spittle) { messaging.convertAndSend("/topic/spittlefeed", spittle); Matcher matcher = pattern.matcher(spittle.getMessage()); if (matcher.find()) { String username = matcher.group(1); // 發送提醒給用戶. messaging.convertAndSendToUser( username, "/queue/notifications", new Notification("You just got mentioned!")); } } } 【5】處理消息異常 1)intro:我們也可以在 控制器方法上添加 @MessageExceptionHandler 注解,讓它來處理 @MessageMapping 方法所拋出的異常; 2)看個荔枝:它會處理 消息方法所拋出的異常; @MessageExceptionHandler public void handleExceptions(Throwable t) { logger.error("Error handling message: " + t.getMessage()); } 3)我們也可以以 參數的形式聲明它所能處理的異常; @MessageExceptionHandler(SpittleException.class) // highlight line. public void handleExceptions(Throwable t) { logger.error("Error handling message: " + t.getMessage()); } // 或者: @MessageExceptionHandler( {SpittleException.class, DatabaseException.class}) // highlight line. public void handleExceptions(Throwable t) { logger.error("Error handling message: " + t.getMessage()); } 4)該方法還可以回應一個錯誤: @MessageExceptionHandler(SpittleException.class) @SendToUser("/queue/errors") public SpittleException handleExceptions(SpittleException e) { logger.error("Error handling message: " + e.getMessage()); return e; } // 如果拋出 SpittleException 的話,將會記錄這個異常,并將其返回. // 而 UserDestinationMessageHandler 會重新路由這個消息到特定用戶所對應的 唯一路徑;

總結

以上是生活随笔為你收集整理的springmvc(18)使用WebSocket 和 STOMP 实现消息功能的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

中文字幕超清在线免费 | 一区二区中文字幕在线观看 | 亚洲激情一区二区三区 | 黄色网www | 国产黄色片免费在线观看 | 久久久久久国产精品999 | 久久中文字幕在线视频 | 九九九九精品 | 美女视频黄,久久 | 色国产精品 | 久久久久国产精品免费免费搜索 | 欧美精品在线一区二区 | 日日夜夜免费精品 | 99精品视频在线观看视频 | 香蕉网在线 | 日日爱av | 99在线视频免费观看 | 国产精品一区二区av麻豆 | 欧美一区二区在线免费看 | 人人爽人人做 | 999亚洲国产996395 | 久久视频在线观看中文字幕 | 日韩免费在线视频 | 成人黄色小说网 | av 一区二区三区 | 伊人五月| 久久a v视频 | 欧美黄色高清 | 亚洲男男gaygay无套 | 欧美 亚洲 另类 激情 另类 | 亚洲一级电影在线观看 | 成人免费亚洲 | 91天天操 | 深夜成人av| 麻豆视频国产 | 夜夜躁狠狠躁日日躁视频黑人 | 国产一卡二卡四卡国 | www.色午夜| 成年人免费看的视频 | 亚洲精品久久久久中文字幕m男 | 99在线观看免费视频精品观看 | 国产精品18久久久久久久 | 在线高清av| 国产精品久久久久久久久久免费看 | 亚洲欧美国产精品18p | 国产人在线成免费视频 | 欧美狠狠操 | 亚洲国产mv| 色综合五月 | 四虎在线观看精品视频 | 日本在线观看一区二区 | 精品在线视频观看 | av片一区二区 | 日本久久免费视频 | 天天躁日日躁狠狠 | 国产97在线观看 | 成人精品福利 | 免费视频三区 | a成人v在线 | 99精品欧美一区二区三区 | 99精品欧美一区二区三区 | 日p视频| 亚洲免费在线看 | 国产精品手机在线观看 | 天天干天天草天天爽 | 69精品 | 三级午夜片 | 国产96在线观看 | 99r在线播放 | 五月视频 | 中文字幕专区高清在线观看 | 丁香电影小说免费视频观看 | 黄网站免费久久 | 亚洲精品视频二区 | 日本一区二区三区视频在线播放 | www.色在线| 婷婷久久婷婷 | 国产永久免费观看 | 国产成人一区二 | 一区在线免费观看 | 超碰在线观看99 | 天堂av高清 | 亚洲成av| 久久99国产一区二区三区 | www天天干 | 四虎影视成人永久免费观看亚洲欧美 | 久久不射电影网 | 国产精品久久久久一区二区三区共 | 日韩精品视频免费在线观看 | 午夜精品区 | 国产精品欧美激情在线观看 | 中文字幕久久精品一区 | 亚洲二区精品 | 国产精品99蜜臀久久不卡二区 | 综合久久综合久久 | 欧美久久久久久久久 | 91人人干| 亚洲mv大片欧洲mv大片免费 | 久久在视频 | 日韩精品久久中文字幕 | 黄色影院在线免费观看 | 国产一级精品绿帽视频 | 日本韩国在线不卡 | 久久精品五月 | 日韩一区二区三区免费视频 | 国产99爱 | 九九视频这里只有精品 | 激情综合网五月激情 | wwwwww黄| 国产高清免费 | 国产精品永久免费视频 | 中文字幕 国产专区 | 久久精品亚洲国产 | 国模一区二区三区四区 | 久久无码av一区二区三区电影网 | 日韩高清不卡一区二区三区 | 成人免费亚洲 | 一级欧美一级日韩 | av黄色影院 | www.久久色.com | 精品一区二区在线观看 | 少妇搡bbbb搡bbb搡69 | 国产黄色网 | 激情综合站 | 精品视频一区在线观看 | 中文字幕一区二区在线播放 | 91麻豆看国产在线紧急地址 | 国产成人99久久亚洲综合精品 | 欧美日本三级 | 六月丁香六月婷婷 | 992tv人人网tv亚洲精品 | 久久久久久久久久久高潮一区二区 | 99国产在线视频 | 午夜久久久精品 | 九九免费在线视频 | 国产a精品 | 91精品少妇偷拍99 | 成人avav| 国产精品免费观看国产网曝瓜 | 成人av.com| 久久免费精品一区二区三区 | 日韩在线观看 | 久久综合九色 | 欧美久久久影院 | 精品久久1 | 蜜臀av性久久久久av蜜臀三区 | 成人资源在线播放 | 在线亚洲日本 | 欧美一级视频一区 | 久草综合在线观看 | 国产一区国产精品 | 涩涩在线 | 狠狠色伊人亚洲综合网站野外 | 黄在线| 精品产品国产在线不卡 | 欧美午夜精品久久久久久浪潮 | 国产高清不卡 | 亚洲国产精品va在线看黑人 | 99视频精品免费视频 | 国产高清亚洲 | 久久精品看片 | 久久av免费电影 | 国产黄色片久久久 | 国产精品美女久久久久久久久久久 | 在线精品视频免费播放 | 国产裸体bbb视频 | 婷婷婷国产在线视频 | 亚洲婷婷综合色高清在线 | 精品一二三四在线 | 久久公开免费视频 | 91视视频在线直接观看在线看网页在线看 | 三级在线国产 | 中文字幕电影网 | 欧美日韩中文在线视频 | 天天干天天干天天射 | 亚洲最新精品 | 成人超碰在线 | 天天艹天天 | 亚洲日本在线视频观看 | 成人在线观看免费 | www.久久视频| 一级黄色电影网站 | 日本精品va在线观看 | 91中文在线观看 | 成人v| 又黄又爽又无遮挡的视频 | 激情一区二区三区欧美 | 亚洲欧洲精品视频 | 国产成年免费视频 | 在线观看激情av | 国产一级在线视频 | 综合精品在线 | 亚洲在线免费视频 | 超碰在线99 | 久久久久免费观看 | 91亚洲精品国偷拍 | 97日日碰人人模人人澡分享吧 | 国产精品久久久久久婷婷天堂 | 色婷婷视频在线观看 | 久久亚洲综合色 | 精品日韩视频 | 久久久久亚洲精品国产 | 亚洲国产美女精品久久久久∴ | av在线播放网址 | 日韩精品久久中文字幕 | www中文在线| 成人黄色电影免费观看 | 激情久久伊人 | 国产在线 一区二区三区 | 亚洲日日射 | 久久久国产99久久国产一 | 91传媒免费在线观看 | 亚洲成年人av | 午夜精品久久久久久久久久久久久久 | 国产精品一区二区吃奶在线观看 | 国产精品大片在线观看 | 成人91av| 久久线视频 | 黄色www| 久久精品一区二区国产 | 99久久精品国产一区 | 亚洲国产精品小视频 | 国产精品久久久久久婷婷天堂 | 精品超碰 | 波多野结衣一区三区 | 四虎影视4hu4虎成人 | 久久久国产精品人人片99精片欧美一 | 中文字幕在线第一页 | 色多多视频在线观看 | 99精品福利 | 九九九九热精品免费视频点播观看 | 欧美另类xxx | 国产日韩精品一区二区在线观看播放 | 成人h在线播放 | 国产伦精品一区二区三区… | 亚洲午夜久久久久久久久电影网 | 深夜免费福利视频 | 99在线免费视频观看 | 国产在线观看,日本 | 欧美性久久久久久 | 中文字幕在线免费播放 | 狠狠狠狠狠狠狠狠 | 伊人中文字幕在线 | 99精品视频观看 | 成人高清av在线 | 欧美另类色图 | 国产精久久久久久久 | 激情欧美一区二区三区 | 日韩在线观看中文 | 人操人| 日韩欧美在线视频一区二区 | 国产视频网站在线观看 | 日韩av网址在线 | 97操操操 | 婷婷六月中文字幕 | 久久丁香 | 久久精品网站视频 | 18做爰免费视频网站 | www.91av在线 | 国产91aaa| 日本久久精品 | 91高清完整版在线观看 | 中文字幕在线有码 | 天天干天天操天天爱 | 日韩理论在线视频 | 国产精品99免费看 | 国产女v资源在线观看 | 精品国产一区二区三区久久久蜜臀 | 国产精品6| 激情五月伊人 | 久久国语露脸国产精品电影 | 免费视频a | 日韩电影在线视频 | 狠狠操欧美 | 亚洲综合激情网 | 国产精品69av| 欧美黑人xxxx猛性大交 | 亚洲精品午夜国产va久久成人 | 国产黄色一级片在线 | 波多野结衣在线观看视频 | 成年人免费在线观看网站 | 欧美视频在线观看免费网址 | 日韩一区二区三区免费视频 | 91av视频在线观看 | 亚洲精选视频免费看 | 91香蕉国产| 国产亚洲精品日韩在线tv黄 | 99久久精品久久久久久动态片 | 在线视频18在线视频4k | 欧美日韩国产精品一区二区三区 | 五月天综合婷婷 | 又黄又爽又色无遮挡免费 | 亚洲精品乱码久久久久久蜜桃动漫 | 国产三级午夜理伦三级 | 午夜精品一二三区 | 日韩亚洲在线视频 | 亚洲天堂香蕉 | 天堂网av在线 | 99精品免费视频 | 久久人网 | 99综合电影在线视频 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 中文字幕在线观看第一区 | 日韩av电影手机在线观看 | 亚洲伦理一区 | 亚洲,播放 | 欧美国产高清 | av在线网站大全 | 欧美另类交在线观看 | 毛片1000部免费看 | av在线直接看 | 丁香六月中文字幕 | 欧美动漫一区二区三区 | 狠狠操电影网 | 在线观看香蕉视频 | 国产高清小视频 | 国产视频日韩视频欧美视频 | 国产精品99久久99久久久二8 | 精品免费一区 | 美女黄网站视频免费 | 九九热视频在线 | 99热最新网址| 国产精品免费观看网站 | 人人澡av | 国产一区免费在线观看 | 在线观看日韩视频 | 国产精品亚洲综合久久 | 丁香婷婷电影 | 婷婷去俺也去六月色 | 91在线九色 | 久久精品牌麻豆国产大山 | 日本在线观看视频一区 | 国产精品尤物 | 超碰公开在线观看 | 久久99热这里只有精品 | 久久电影中文字幕视频 | 国产精品美女久久久久久久久久久 | 成年人免费看片网站 | 精品欧美小视频在线观看 | 狠狠干婷婷色 | 中文亚洲欧美日韩 | 亚洲综合一区二区精品导航 | 亚洲精品国产精品国自产 | 91av片 | 麻豆免费观看视频 | 中文国产在线观看 | 天天干,夜夜爽 | 国产精品密入口果冻 | 国产精品中文字幕av | 国产高清免费观看 | 日韩欧美一区二区三区在线 | 麻豆91精品视频 | 中文字幕在线观看三区 | 亚洲春色综合另类校园电影 | 国产精品麻豆一区二区三区 | 久久国产亚洲视频 | 一区二区欧美在线观看 | 美女网站色在线观看 | 人人爽人人片 | 国产欧美综合视频 | 麻豆手机在线 | 日韩xxxx视频| 午夜视频在线观看一区二区 | 亚洲精品乱码久久久久久9色 | 丁香花在线观看视频在线 | av 一区 二区 久久 | 中文字幕在线观看视频一区二区三区 | 99精品在线免费观看 | 精品久久久久久久久久久久久久久久久久 | 免费看毛片网站 | 久久国产精品二国产精品中国洋人 | 在线观看视频中文字幕 | 日日综合 | 91免费观看视频网站 | 成人av在线资源 | 综合色中文 | 青草视频在线免费 | 视频三区 | 69av视频在线| 亚洲精品www | 国产黄网站在线观看 | 国产精品久久久久一区二区 | 91手机视频 | 欧美日韩精品网站 | 国产九九九视频 | 亚洲日韩欧美一区二区在线 | 日韩中文在线字幕 | 91一区二区三区久久久久国产乱 | 日韩欧美99 | 五月色丁香 | 国产一区二区三区午夜 | 国产精品18久久久久久久网站 | 免费在线一区二区 | 丁香五月缴情综合网 | 国产精品一区二区视频 | 国产爽视频 | 日日碰夜夜爽 | 久久精品视频18 | 激情综合网五月婷婷 | 精品免费| 国产高清黄色 | 日本一区二区三区视频在线播放 | 亚洲色综合 | 久久免费一级片 | 午夜资源站 | 岛国大片免费视频 | 少妇av网 | 在线观看的黄色 | 国产成人91 | 亚洲精品一区二区久 | 99精品视频播放 | 国产精品美女久久久久久 | 成人精品亚洲 | h视频在线看 | 97超碰在线久草超碰在线观看 | 国产一及片 | 亚洲精品国产精品国自产 | 娇妻呻吟一区二区三区 | 视频一区二区三区视频 | 天堂av网址| 69亚洲视频 | 国产又黄又硬又爽 | 97色涩| 国产精品视频在线看 | 国语久久 | 国产精品午夜在线观看 | 超碰在线人人艹 | 狠狠狠综合 | 久久久噜噜噜久久久 | 一区二区av | av免费成人 | 亚洲人成网站精品片在线观看 | 免费在线激情电影 | 国产成人精品一区二区在线 | 樱空桃av| 亚洲一级片免费观看 | 国产美女无遮挡永久免费 | a v在线视频| 欧美另类xxx| 亚洲一区二区三区在线看 | 色综合久久88色综合天天免费 | 国产99久久久国产精品成人免费 | 亚洲一二三区精品 | 天天玩天天干 | 开心色停停 | 中文免费在线观看 | 手机成人免费视频 | 午夜精品久久久久久久久久 | 亚洲欧美视频 | 男女激情麻豆 | 最近中文字幕视频完整版 | 88av视频| 五月婷婷播播 | 成人蜜桃网 | 成人一级在线观看 | 天天综合视频在线观看 | 免费在线观看日韩视频 | 久草手机视频 | 久久免费视频这里只有精品 | 久久久黄色av | 成人资源在线 | 国产精品日韩欧美 | 久久久久久久久久伊人 | 美腿丝袜一区二区三区 | 国产成人黄色在线 | 久久se视频| 91网址在线观看 | 毛片二区 | 久操视频在线播放 | 亚洲爱爱视频 | 99久久婷婷国产综合精品 | 91av国产视频 | 国产视频2 | 操操操天天操 | 天堂网av 在线 | www免费视频com━ | 日韩一二三 | 久产久精国产品 | 韩国av不卡| 成人亚洲精品久久久久 | 久久麻豆精品 | 亚洲成人黄色 | 日韩在线观 | 伊人网av| 又紧又大又爽精品一区二区 | 精品免费国产一区二区三区四区 | 国精产品999国精产 久久久久 | 国产免费中文字幕 | 婷婷色在线视频 | 香蕉视频在线视频 | 在线观看911视频 | 国产在线最新 | 久草免费在线观看 | 久久久电影 | 国产成人精品亚洲a | 四虎国产永久在线精品 | 9在线观看免费 | 色视频在线免费观看 | 国产高清视频免费最新在线 | 88av视频| 亚洲视频在线观看 | 免费毛片aaaaaa | 人人爽人人爱 | 97在线观看免费观看 | 免费视频成人 | 亚洲天堂在线观看完整版 | 亚洲国产黄色 | 久久精品视频3 | 草久久久久 | 中文字幕在线视频一区二区 | av高清网站在线观看 | 午夜 免费| 青青河边草免费直播 | 97狠狠操 | 香蕉在线视频播放网站 | 精品国产一区二区三区日日嗨 | 欧美日韩aa | 精品欧美日韩 | 中文字幕电影网 | 国产精品中文字幕av | 日日爽视频 | 中文字幕在线视频一区二区 | 精品国产诱惑 | 欧美精品天堂 | 最新av免费在线 | 日韩二级毛片 | 99久久久久成人国产免费 | 亚洲 欧美 变态 国产 另类 | 国产一级片播放 | 色婷婷综合久久久中文字幕 | 人人干天天干 | 久久99国产综合精品 | 免费高清影视 | 丁香花中文字幕 | 91九色蝌蚪视频在线 | 日本爱爱免费视频 | 久久久久久久久久久精 | 久久99久久99精品免视看婷婷 | 成人黄色中文字幕 | 777视频在线观看 | 九九热在线观看视频 | 国产精品一区二区在线播放 | 亚洲精品女 | 天天操天天爽天天干 | 欧美激情精品一区 | 亚洲精品视频在线免费 | 日日夜夜草 | 日韩经典一区二区三区 | 精品国产自在精品国产精野外直播 | 91成人免费看| 四虎国产精品永久在线国在线 | 国产精品va在线播放 | 国产香蕉av | 久久精品一区二区三区国产主播 | 人人澡人摸人人添学生av | 国产色在线观看 | 成人免费在线网 | 日操干| 国产在线a视频 | 久久精品视频国产 | 高清美女视频 | 97超级碰碰 | 亚洲欧美日韩国产一区二区三区 | 日日麻批40分钟视频免费观看 | 丁香网婷婷 | 狠狠干综合 | 国产精品综合在线 | 91久草视频 | 在线观看小视频 | 欧美视频www | 色婷婷狠狠五月综合天色拍 | 国产高清不卡av | 国产在线高清 | 久久久亚洲精品 | 久久免费在线视频 | 91精品国自产在线观看欧美 | www久久久久| 国产黑丝袜在线 | 在线观看色网站 | 鲁一鲁影院 | 一级一片免费看 | 国产一区在线观看视频 | 久久高清片 | 夜夜躁日日躁狠狠久久88av | 中文字幕有码在线 | 日韩精品欧美精品 | 亚洲精品麻豆 | 欧美地下肉体性派对 | av黄色大片 | 夜夜躁天天躁很躁波 | 九草在线观看 | 午夜丁香视频在线观看 | 久久人人爽视频 | 国产精品久久久久永久免费看 | 亚洲精品乱码久久久久 | 亚洲精品视频网站在线观看 | 日韩夜夜爽 | 国产精品久久久久一区 | 国产成人精品在线播放 | 美女久久网站 | 亚洲在线观看av | 麻豆视频在线 | 在线免费观看涩涩 | 蜜臀av性久久久久蜜臀aⅴ四虎 | 免费在线观看毛片网站 | 日韩免费在线观看视频 | 日韩精品aaa | 亚洲国产中文在线观看 | 色视频网站在线观看一=区 a视频免费在线观看 | 久久久久看片 | 国产一区在线视频 | 999久久国产精品免费观看网站 | 久久婷婷五月综合色丁香 | 97精品国产一二三产区 | 国产96在线 | 6080yy精品一区二区三区 | 最新av电影网址 | 免费的国产精品 | 超碰999 | 日韩一级片网址 | 亚洲精品免费在线观看视频 | 天天操夜夜拍 | 国产美女免费 | av大全免费在线观看 | 中文字幕资源站 | av在线免费在线 | 免费看黄网站在线 | 在线观看黄a | 国产a精品 | 91成人破解版 | 婷婷日日| 欧美巨大 | 不卡的av电影 | 久草在线国产 | 欧美另类69 | 三级性生活视频 | 精品久久影院 | 91亚洲激情 | 亚洲精品久久久久久国 | 亚洲最大免费成人网 | 日韩中文字幕在线 | 美女免费视频网站 | 国产在线欧美日韩 | 婷婷国产在线 | 黄色特一级片 | 最近中文字幕在线中文高清版 | 久久久精品小视频 | 久久高视频 | 丁香六月婷婷 | 亚洲综合色站 | 国产一区影院 | 精品国产成人在线 | 嫩模bbw搡bbbb搡bbbb | 国产成人精品av在线观 | 亚洲精品色 | 亚洲精品视频免费在线观看 | 国产色秀视频 | 精品专区一区二区 | 全久久久久久久久久久电影 | 一区二区三区精品在线视频 | 久久久久国产精品一区 | 国产亚洲精品久久久久久久久久 | 午夜私人影院 | 国产精品女人久久久 | 天天做天天爱天天爽综合网 | 色综合久久综合中文综合网 | 国产h片在线观看 | 国产视频91在线 | 天天天干 | 在线а√天堂中文官网 | 岛国大片免费视频 | 日韩欧美不卡 | 国产精品综合在线观看 | 色综合国产 | 久久久久福利视频 | 国产自偷自拍 | 国产自在线观看 | 黄色大全免费观看 | 免费观看性生交大片3 | 色av网站 | av福利在线免费观看 | 国产黄色视 | 亚洲一区视频免费观看 | 国产精品一区二区免费在线观看 | 麻豆av电影 | 国产午夜亚洲精品 | 香蕉视频在线免费看 | 91久久国产综合精品女同国语 | 香蕉在线视频播放网站 | 天天干天天操天天干 | 中文一二区 | 91av中文字幕| 天天射天天拍 | 99久久综合国产精品二区 | 日本中文一区二区 | 亚洲黄色成人 | 欧美日韩亚洲在线 | 黄色一级免费网站 | 亚洲成人精品av | 久久短视频 | 日本精品视频一区二区 | 国产免码va在线观看免费 | 麻豆高清免费国产一区 | 国产精品色婷婷视频 | 中文字幕高清av | 在线观看亚洲电影 | 人人干人人添 | 伊人伊成久久人综合网小说 | 最近中文字幕高清字幕免费mv | 久久精品视频日本 | 久久久久中文字幕 | 91精品视频免费观看 | 日韩另类在线 | 99久久久久国产精品免费 | 午夜999 | 西西人体4444www高清视频 | 亚洲视频免费在线观看 | 久久久综合色 | 免费特级黄毛片 | 中文字幕日本电影 | 四虎影视8848dvd | 麻豆影视网站 | 国产精品久久久久久久久久免费看 | 五月开心婷婷 | 国产视频在线观看一区 | 国内小视频在线观看 | 色综合久久88色综合天天6 | 人交video另类hd | 在线免费中文字幕 | 国产精品国产亚洲精品看不卡 | 国产一级小视频 | 欧美极品少妇xbxb性爽爽视频 | 密桃av在线 | 福利视频在线看 | 日韩美精品视频 | 日韩经典一区二区三区 | 一级黄视频 | 欧美极品一区二区三区 | 国产在线观看免费观看 | 九九热在线观看 | av福利资源| 婷婷五月在线视频 | 日韩中文字幕免费视频 | 涩涩网站在线观看 | 91在线国内视频 | 国产日韩中文在线 | 亚洲人在线视频 | www中文在线 | 亚州欧美视频 | 人人超碰97 | 天天综合色 | 在线激情小视频 | 最近免费中文字幕大全高清10 | 日韩中文字幕网站 | av超碰在线观看 | 国产精品va在线播放 | 91精品视频在线 | 狠狠色丁香久久婷婷综合_中 | 最新高清无码专区 | 国产在线欧美在线 | 看黄色.com| 午夜精品久久久久99热app | 婷婷在线免费观看 | 日韩精品中文字幕av | 婷婷色综合色 | 999久久久免费视频 午夜国产在线观看 | 亚洲精品无 | 日韩综合在线观看 | 精品产品国产在线不卡 | 日韩精品91偷拍在线观看 | 亚洲欧美日韩在线一区二区 | 五月婷婷一区二区三区 | 国产精品亚洲a | 天天操天天射天天舔 | 久久精品成人热国产成 | www黄在线| 国产一区二区三区久久久 | 午夜91视频 | 国产资源精品在线观看 | 欧美一级裸体视频 | 精品久久网 | 99国产视频 | 麻豆综合网 | 久久久久久久久久久精 | 91视频 - x99av| 992tv在线观看| 安徽妇搡bbbb搡bbbb | 九九九热精品免费视频观看网站 | 在线不卡中文字幕播放 | 一区精品在线 | 国产精品久久久 | 欧美视频www| 日韩一区二区三区免费电影 | 国产在线色视频 | 亚洲黄网站 | 99久久精品国产毛片 | 九九九九九九精品任你躁 | 久久视频免费在线观看 | 成人免费在线网 | 国产一区二区视频在线 | 亚洲综合色站 | 91干干干| 亚洲激情 在线 | 九色在线| 国产精品高清免费在线观看 | 五月天激情婷婷 | 色婷婷国产精品一区在线观看 | 激情av网址| 国产午夜一级毛片 | 探花视频免费观看 | 国产精品久久久久一区二区国产 | 久久精品免费看 | 午夜在线观看一区 | 9999精品视频 | 国产精品一区二区三区观看 | 夜夜澡人模人人添人人看 | 久久久久久国产一区二区三区 | 国产精品手机视频 | 中文字幕日韩国产 | 中文字幕在线观看完整版电影 | 激情久久久久久久久久久久久久久久 | 91av在线视频免费观看 | 国产麻豆精品久久一二三 | 黄色在线视频网址 | 亚洲国产日韩在线 | 四虎成人免费影院 | 亚洲黄色在线播放 | 国产91免费看 | 日日夜夜狠狠 | 久久综合精品一区 | 成人免费在线观看av | 福利电影久久 | 91高清视频免费 | 国产精品永久久久久久久www | 在线视频观看亚洲 | 久久亚洲精品电影 | a午夜在线| 狠狠色狠狠色综合日日小说 | 中文字幕永久 | 欧美成人91 | 国产自产高清不卡 | 亚洲成人频道 | 伊人久久电影网 | 欧美日韩高清在线 | 国产成人久久久久 | 最新av中文字幕 | 久久久久免费电影 | 蜜臀久久99静品久久久久久 | 天天色天天上天天操 | 日韩视频免费 | 日韩欧美在线综合网 | 国产我不卡 | 99视频精品全部免费 在线 | 97碰碰视频| 日韩欧美视频一区二区三区 | 91久久人澡人人添人人爽欧美 | 国产精品久久久777 成人手机在线视频 | 91成年视频| 国产第一福利网 | 午夜黄色影院 | 日本3级在线观看 | 日韩视频中文 | 午夜婷婷网 | 2023年中文无字幕文字 | 天天干天天射天天插 | 超碰在线98 | 97精品国自产拍在线观看 | 日韩精品观看 | 久久综合九色综合97婷婷女人 | 九九视频热 | 精品国产伦一区二区三区观看说明 | 欧美与欧洲交xxxx免费观看 | 欧美一区二区日韩一区二区 | 天天操天天色综合 | 国产成年人av | 亚洲国产精品va在线看黑人动漫 | 久久成人免费 | 亚洲黄色成人网 | 日韩电影在线一区 | 久草久草在线 | 免费观看丰满少妇做爰 | 国产精品亚洲片夜色在线 | 欧美成人精品欧美一级乱 | 亚洲乱码久久 | 中文字幕在线观看你懂的 | 亚洲伊人第一页 | 亚洲精品乱码久久久久v最新版 | 天天干一干| 高清免费在线视频 | 国产网站在线免费观看 | 在线看成人 | 中文字幕在线日亚洲9 | 国产美女视频黄a视频免费 久久综合九色欧美综合狠狠 | 日韩综合一区二区三区 | 在线黄网站 | 黄色一级在线观看 | 黄色毛片大全 | 九月婷婷综合网 | 超碰97在线资源 | 丁香六月婷 | 91精品国产成 | 美女搞黄国产视频网站 | 亚洲一级性 | 免费a一级| 一区二区亚洲精品 | 欧美一级淫片videoshd | 久久夜视频 | 色综合天天狠狠 | 国产精品成人在线观看 | 欧美日韩成人一区 | 亚洲高清视频在线观看免费 | 99久高清在线观看视频99精品热在线观看视频 | 国产亚洲成av片在线观看 | 国产精品福利久久久 | 久久久男人的天堂 | 成年人免费看片网站 | 中文字幕你懂的 | av黄色在线播放 | 在线亚洲欧美视频 | 久久久久久久久久久国产精品 | 黄色com | 国产成人精品一区二三区 | 免费av网站在线看 | 色婷丁香 | 亚洲视频在线播放 | 网站免费黄 | 黄色tv视频 | 黄色官网在线观看 | 免费涩涩网站 | 天天操夜夜曰 | 国产精品二区三区 | 在线a亚洲视频播放在线观看 | 99视频国产精品免费观看 | 激情欧美xxxx | 天天爽人人爽夜夜爽 | 玖玖视频 | 西西大胆免费视频 | 伊人色**天天综合婷婷 | 高清av网| 久久夜av| 国产免费一区二区三区网站免费 | 久久精品香蕉视频 | 久久久成人精品 | 国产伦精品一区二区三区高清 | 久久99精品国产99久久 | 色com | 天天天天天操 | 国产精品成人一区二区三区吃奶 | 国产精品自在线 | 麻豆视频观看 | 六月激情 | 国产一级视频免费看 | 日韩特级片 | 综合天堂av久久久久久久 | 免费午夜在线视频 | 国产精品一区二区久久久 | 99日韩精品 | 91视频a | 成人精品在线 | 伊人影院99 | 免费在线观看一级片 | 久久久精品综合 | 91精选在线观看 | 中文字幕av在线 | 激情欧美在线观看 | 亚州天堂| 视频国产区 | 美女福利视频 | 狠狠操欧美 | 亚洲精品在线网站 | 天天拍天天操 | 国产精品久久久久久久久久免费 | 免费看v片网站 | 久久久国产在线视频 | 天堂在线视频中文网 | 久久久久久久久久网站 | 九九视频热 | 中日韩欧美精彩视频 | 手机看片国产 | 成人免费观看在线视频 | 日本精品va在线观看 | 在线看片日韩 | 一区二区三区免费在线观看视频 | 亚洲精品1区2区3区 超碰成人网 | 日韩在线观看不卡 | 精品国产一区二区三区久久久 | 欧美日韩亚洲精品在线 | 国产3p视频| 亚洲不卡123 | 久久久www| 成人av免费在线看 | 国产精品久久久久一区 | 久久久久成人精品免费播放动漫 | 日韩av电影网站在线观看 | 九九免费在线观看视频 | 91丨九色丨国产在线观看 | 精品播放| 五月婷av |