什么是WebSocket?
WebSocket协议是基于TCP的一种新的网络协议。它实现了浏览器与服务器全双工(full-duplex)通信——允许服务器主动发送信息给客户端。
为什么需要 WebSocket?
初次接触 WebSocket 的人,都会问同样的问题:我们已经有了 HTTP 协议,为什么还需要另一个协议?它能带来什么好处?
- 答案很简单,因为 HTTP 协议有一个缺陷:通信只能由客户端发起,HTTP 协议做不到服务器主动向客户端推送信息。
举例来说,我们想要查询当前的排队情况,只能是页面轮询向服务器发出请求,服务器返回查询结果。轮询的效率低,非常浪费资源(因为必须不停连接,或者 HTTP 连接始终打开)。因此WebSocket 就是这样发明的。
Maven依赖
1 <!--springboot的高级组件会自动引用基础的组件-->
2 <!--,像spring-boot-starter-websocket就引入了spring-boot-starter-web和spring-boot-starter,-->
3 <!--所以不要重复引入。-->
4 <!--JavaEE标准-->
5 <dependency>
6 <groupId>javax</groupId>
7 <artifactId>javaee-api</artifactId>
8 <scope>provided</scope>
9 </dependency>
10 <!--websocket-->
11 <dependency>
12 <groupId>org.springframework.boot</groupId>
13 <artifactId>spring-boot-starter-websocket</artifactId>
14 </dependency>
WebSocket配置类
1 package com.zr.demo.config;
2
3 import org.springframework.context.annotation.Bean;
4 import org.springframework.context.annotation.Configuration;
5 import org.springframework.web.socket.server.standard.ServerEndpointExporter;
6
7 /**
8 * @Author: Hujh
9 * @Date: 2019/7/16 11:10
10 * @Description: WebSocket配置类
11 */
12 @Configuration
13 public class WebSocketConfig {
14
15 /*使用@ServerEndpoint创立websocket endpoint*/
16 @Bean
17 public ServerEndpointExporter serverEndpointExporter() {
18 return new ServerEndpointExporter();
19 }
20 }
WebSocketServer
因为WebSocket是类似客户端服务端的形式(采用ws协议),那么这里的WebSocketServer其实就相当于一个ws协议的Controller直接@ServerEndpoint("/websocket")@Component启用即可,然后在里面实现@OnOpen,@onClose,@onMessage等方法
1 package com.zr.demo.socket;
2
3 import org.springframework.stereotype.Component;
4
5 import javax.websocket.OnClose;
6 import javax.websocket.OnMessage;
7 import javax.websocket.OnOpen;
8 import javax.websocket.Session;
9 import javax.websocket.server.ServerEndpoint;
10 import java.io.IOException;
11 import java.util.concurrent.CopyOnWriteArraySet;
12
13 /**
14 * @Author: Hujh
15 * @Date: 2019/7/16 11:13
16 * @Description: webSocket实现类
17 */
18 @ServerEndpoint(value = "/webSocket")
19 @Component
20 public class WebSocket {
21 //静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
22 private static int onlineCount = 0;
23
24 //concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。
25 private static CopyOnWriteArraySet<WebSocket> webSocketSet = new CopyOnWriteArraySet<WebSocket>();
26
27 //与某个客户端的连接会话,需要通过它来给客户端发送数据
28 private Session session;
29
30 /**
31 * 连接建立成功调用的方法
32 */
33 @OnOpen
34 public void onOpen(Session session) {
35 this.session = session; //加入set中
36 webSocketSet.add(this);
37 addOnlineCount(); //在线数加1
38 System.out.println("有新连接加入!当前在线人数为:" + getOnlineCount());
39 try {
40 sendMessage("连接成功");
41 } catch (IOException e) {
42 System.out.println("IO异常");
43 }
44 }
45
46 /**
47 * 连接关闭调用的方法
48 */
49 @OnClose
50 public void onClose() {
51 webSocketSet.remove(this); //从set中删除
52 subOnlineCount(); //在线数减1
53 System.out.println("有一连接关闭!当前在线人数为" + getOnlineCount());
54 }
55
56 /**
57 * 收到客户端消息后调用的方法
58 * @param message 客户端发送过来的消息
59 */
60 @OnMessage
61 public void onMessage(String message, Session session) {
62 System.out.println("来自客户端的消息:" + message);
63
64 //群发消息
65 for (WebSocket item : webSocketSet) {
66 try {
67 item.sendMessage(message);
68 } catch (IOException e) {
69 e.printStackTrace();
70 }
71 }
72 }
73
74 /**
75 * 发生错误时调用
76 @OnError
77 */
78 public void onError(Session session, Throwable error) {
79 System.out.println("发生错误");
80 error.printStackTrace();
81 }
82
83 /*
84 * @Title: sendMessage
85 * @Author : Hujh
86 * @Date: 2019/7/17 10:52
87 * @Description : 发送消息
88 * @param : message
89 * @Return : void
90 */
91 public void sendMessage(String message) throws IOException {
92 this.session.getBasicRemote().sendText(message);
93 //this.session.getAsyncRemote().sendText(message);
94 }
95
96
97 /*
98 * @Title: sendInfo
99 * @Author : Hujh
100 * @Date: 2019/7/17 10:52
101 * @Description : 群发自定义消息
102 * @param : message
103 * @Return : void
104 */
105 public static void sendInfo(String message) throws IOException {
106 for (WebSocket item : webSocketSet) {
107 try {
108 item.sendMessage(message);
109 } catch (IOException e) {
110 continue;
111 }
112 }
113 }
114
115 /*
116 * @Title: getOnlineCount
117 * @Author : Hujh
118 * @Date: 2019/7/17 10:51
119 * @Description : 获得当前在线人数
120 * @param :
121 * @Return : int
122 */
123 public static synchronized int getOnlineCount() {
124 return onlineCount;
125 }
126
127 /*
128 * @Title: addOnlineCount
129 * @Author : Hujh
130 * @Date: 2019/7/17 10:51
131 * @Description : 在线人数+1
132 * @param :
133 * @Return : void
134 */
135 public static synchronized void addOnlineCount() {
136 WebSocket.onlineCount++;
137 }
138
139 /*
140 * @Title: subOnlineCount
141 * @Author : Hujh
142 * @Date: 2019/7/17 10:52
143 * @Description :在线人数-1
144 * @param :
145 * @Return : void
146 */
147 public static synchronized void subOnlineCount() {
148 WebSocket.onlineCount--;
149 }
150
151 }
消息推送
至于推送新信息,可以再自己的Controller写个方法调用WebSocket.sendInfo();
1 package com.zr.demo.controller;
2
3 import com.zr.demo.socket.WebSocket;
4 import org.springframework.beans.factory.annotation.Autowired;
5 import org.springframework.web.bind.annotation.RequestMapping;
6 import org.springframework.web.bind.annotation.RequestParam;
7 import org.springframework.web.bind.annotation.RestController;
8
9 /**
10 * @Author: Hujh
11 * @Date: 2019/7/19 15:40
12 * @Description: 消息发送控制器
13 */
14 @RestController
15 public class SendMessageController {
16
17 @Autowired
18 private WebSocket webSocket;
19
20 @RequestMapping("sendInfo")
21 public String sendInfo(@RequestParam String msg) {
22 try {
23 webSocket.sendInfo(msg);
24 } catch (Exception e) {
25 e.printStackTrace();
26 return "信息发送异常!";
27 }
28
29 return "发送成功~";
30 }
31
32 }
页面代码
1 <!DOCTYPE HTML>
2 <html>
3 <head>
4 <title>WebSocket</title>
5 </head>
6
7 <body>
8 <div style="width: 800px;height: 100%; margin: 0px auto;">
9 <span style="color: coral; font-size: 22px;"> Welcome WebSocket</span><br/><br/>
10 <div id="message">
11 </div>
12 </div>
13 </body>
14
15 <script type="text/javascript">
16 var websocket = null;
17 //判断当前浏览器是否支持WebSocket
18 if('WebSocket' in window){
19 websocket = new WebSocket("ws://localhost:8082/webSocket");
20 }else{
21 alert('连接失败!!')
22 }
23
24 //连接发生错误的回调方法
25 websocket.onerror = function(){
26 setMessageInnerHTML("error");
27 };
28
29 //连接成功建立的回调方法
30 websocket.onopen = function(event){
31 setMessageInnerHTML("webSocket 连接成功~");
32 }
33
34 //接收到消息的回调方法
35 websocket.onmessage = function(event){
36 setMessageInnerHTML(event.data);
37 }
38
39 //连接关闭的回调方法
40 websocket.onclose = function(){
41 setMessageInnerHTML("close");
42 }
43
44 //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,
45 // 防止连接还没断开就关闭窗口,server端会抛异常。
46 window.onbeforeunload = function(){
47 websocket.close();
48 }
49
50 //将消息显示在网页上
51 function setMessageInnerHTML(innerHTML){
52 document.getElementById('message').innerHTML += innerHTML + '<br/>';
53 }
54
55 //关闭连接
56 function closeWebSocket(){
57 websocket.close();
58 }
59
60 61 </script>
62 </html>
测试效果
1.建立两个连接
2.消息推送
注:本文仅作为个人学习记录,不提供任何参考价值!