一、新建maven工程javascript
工程結構以下:css
完整的pom.xml以下:html
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>com.websocket.demo</groupId> <artifactId>websocket-demo</artifactId> <version>1.0-SNAPSHOT</version> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>1.4.1.RELEASE</version> </parent> <dependencies> <!-- thymeleaf 模板的配置 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-thymeleaf</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- spring websocket的配置 --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-websocket</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-devtools</artifactId> <scope>runtime</scope> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-autoconfigure</artifactId> <version>1.4.5.RELEASE</version> </dependency> </dependencies> <build> <plugins> <plugin> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-maven-plugin</artifactId> </plugin> </plugins> </build> </project>
二、application.properties 的配置以下前端
#thymeleaf start
#spring.thymeleaf.mode=HTML5
spring.thymeleaf.encoding=UTF-8
spring.thymeleaf.content-type=text/html
#開發時關閉緩存,否則無法看到實時頁面
spring.thymeleaf.cache=false
spring.thymeleaf.prefix=classpath:/templates/
#thymeleaf end
server.port=8082
三、啓動類java
package com.websocket; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; /** * @author Administrator * @date 2018/08/18 */ @SpringBootApplication public class Application { public static void main(String[] args) throws Exception { SpringApplication.run(Application.class, args); } }
四、WebsocketConfig 類進行了websocket的配置web
package com.websocket.config; import org.springframework.context.annotation.Configuration; 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; /** * @author hzb * @date 2018/09/30 */ @Configuration @EnableWebSocketMessageBroker public class WebsocketConfig extends AbstractWebSocketMessageBrokerConfigurer { @Override public void configureMessageBroker(MessageBrokerRegistry config) { //啓用/userTest,/topicTest,兩個消息前綴 config.enableSimpleBroker("/userTest","/topicTest"); //若是不設置下面這一句,用convertAndSendToUser來發送消息,前端訂閱只能用/user開頭。 config.setUserDestinationPrefix("/userTest"); //客戶端(html等)向服務端發送消息的前綴 config.setApplicationDestinationPrefixes("/app"); } @Override public void registerStompEndpoints(StompEndpointRegistry stompEndpointRegistry) { //客戶端和服務端進行鏈接的endpoint stompEndpointRegistry.addEndpoint("/websocket-endpoint").setAllowedOrigins("*").withSockJS(); } }
五、消息管理實現類spring
package com.websocket.controller; 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.SimpMessagingTemplate; import org.springframework.scheduling.annotation.EnableScheduling; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.Date; /** * @author hzb * @date 2018/09/30 */ @Controller @EnableScheduling public class WebsocketMsgController { @Autowired private SimpMessagingTemplate messagingTemplate; @GetMapping("/") public String index() { return "index"; } /** * index.html將message發送給後端,後端再將消息重組後發送到/topicTest/web-to-server-to-web * @param message * @return * @throws Exception */ @MessageMapping("/send") @SendTo("/topicTest/web-to-server-to-web") public String send(String message) throws Exception { DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); return "服務器將原消息返回: "+df.format(new Date())+" :" + message; } /** * 最基本的服務器端主動推送消息給前端 * @return * @throws Exception */ @Scheduled(fixedRate = 1000) public String serverTime() throws Exception { // 發現消息 DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); messagingTemplate.convertAndSend("/topicTest/servertime", df.format(new Date())); return "servertime"; } /** * 如下面這種方式發送消息,前端訂閱消息的方式爲: stompClient.subscribe('/userTest/hzb/info' * @return * @throws Exception */ @Scheduled(fixedRate = 1000) public String serverTimeToUser() throws Exception { // 發現消息 DateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); //這裏雖然沒有指定發送前綴爲/userTest,可是在WebsocketConfig.java中設置了config.setUserDestinationPrefix("/userTest"), //不然默認爲/user messagingTemplate.convertAndSendToUser("hzb","/info", df.format(new Date())); return "serverTimeToUser"; } }
前端代碼:index.htmlapache
<!DOCTYPE html> <html> <head> <title>玩轉spring boot——websocket</title> <script src="//cdn.bootcss.com/angular.js/1.5.6/angular.min.js"></script> <script src="https://cdn.bootcss.com/sockjs-client/1.1.4/sockjs.min.js"></script> <script src="https://cdn.bootcss.com/stomp.js/2.3.3/stomp.min.js"></script> <script type="text/javascript"> var stompClient = null; var app = angular.module('app', []); app.controller('MainController', function($rootScope, $scope, $http) { $scope.data = { connected : false, sendMessage : '', receivMessages : [] }; //鏈接 $scope.connect = function() { var socket = new SockJS('/websocket-endpoint'); stompClient = Stomp.over(socket); stompClient.connect({}, function(frame) { // 訂閱後端主動推消息到前端的topic stompClient.subscribe('/topicTest/servertime', function(r) { $scope.data.time = '當前服務器時間:' + r.body; $scope.data.connected = true; $scope.$apply(); }); // 閱後端主動推消息到前端的topic,只有指定的用戶(hzb)收到的的消息 stompClient.subscribe('/userTest/hzb/info', function(r) { $scope.data.hzbtime = '當前服務器時間:' + r.body; $scope.data.connected = true; $scope.$apply(); }); // 訂閱前端發到後臺,後臺又將消息返回前臺的topic stompClient.subscribe('/topicTest/web-to-server-to-web', function(msg) { $scope.data.receivMessages.push(msg.body); $scope.data.connected = true; $scope.$apply(); }); $scope.data.connected = true; $scope.$apply(); }); }; $scope.disconnect = function() { if (stompClient != null) { stompClient.disconnect(); } $scope.data.connected = false; } $scope.send = function() { stompClient.send("/app/send", {}, JSON.stringify({ 'message' : $scope.data.sendMessage })); } }); </script> </head> <body ng-app="app" ng-controller="MainController"> <h2>websocket示例</h2> <label>WebSocket鏈接狀態:</label> <button type="button" ng-disabled="data.connected" ng-click="connect()">鏈接</button> <button type="button" ng-click="disconnect()" ng-disabled="!data.connected">斷開</button> <br/> <br/> <div ng-show="data.connected"> <h4>如下是websocket的服務端主動推送消息到頁面的例子</h4> <label>{{data.time}}</label> <br/> <br/> </div> <div ng-show="data.connected"> <h4>如下是websocket的服務端主動推送消息到頁面的例子,只有hzb這個用戶收到</h4> <label>{{data.hzbtime}}</label> <br/> <br/> </div> <div ng-show="data.connected"> <h4>如下是websocket的客戶端發消息到服務端,服務端再將該消息返回到客戶端(頁面)的例子</h4> <input type="text" ng-model="data.sendMessage" placeholder="請輸入內容..." /> <button ng-click="send()" type="button">發送</button> <br/> <table> <thead> <tr> <th>消息內容:</th> </tr> </thead> <tbody> <tr ng-repeat="messageContent in data.receivMessages"> <td>{{messageContent}}</td> </tr> </tbody> </table> </div> </body> </html>
運行效果後端
沒有點「鏈接」以前緩存
點了「鏈接」以後
輸入內容:點擊發送