SpringBoot 全家桶 | WebSocket服务端与客户端实例

本文源码:Gitee·点这里

介绍

WebSocket 是 HTML5 开始提供的一种在单个 TCP 连接上进行全双工通讯的协议。

WebSocket 使得客户端和服务器之间的数据交换变得更加简单,允许服务端主动向客户端推送数据。在 WebSocket API 中,浏览器和服务器只需要完成一次握手,两者之间就直接可以创建持久性的连接,并进行双向数据传输。

参考

  1. Spring Framework 中文文档

  2. WebSocket 在线测试 v13

  3. HTML5 WebSocket

Server端

Server端我们使用SpringBoot的一个包spring-boot-starter-websocket,我们来引入它

<dependency>
 <groupId>org.springframework.boot</groupId>
 <artifactId>spring-boot-starter-websocket</artifactId>
</dependency>
WebSocket处理器

WebSocket处理器用来处理客户端发送的消息,Spring中常用的是两个TextWebSocketHandlerBinaryWebSocketHandler,我们使用TextWebSocketHandler

import org.springframework.web.socket.handler.TextWebSocketHandler;
​
public class ChatWebSocketHandler extends TextWebSocketHandler {
​
 @Override
 public void afterConnectionEstablished(WebSocketSession session) throws Exception {

 }
​
 @Override
 protected void handleTextMessage(WebSocketSession session, TextMessage message) throws Exception {
 String msg = message.getPayload();
 }

 @Override
 public void handleTransportError(WebSocketSession session, Throwable exception) throws Exception {

 }

 @Override
 public void afterConnectionClosed(WebSocketSession session, CloseStatus status) throws Exception {

 }
}

重写TextWebSocketHandler的四个方法:

  • afterConnectionEstablished 成功创建连接后调用

  • handleTextMessage 收到客户端消息后调用

  • handleTransportError 连接异常时调用

  • afterConnectionClosed 连接关闭后调用

WebSocketSession是客户端与服务端建立的回话,可以通过close()方法主动关闭连接

TextMessage为收到的消息,可以通过getPayload()方法获取消息内容

WebSocket配置

有了处理器了,就可以将此处理器映射到指定path上了,这需要增加一些配置,Spring提供一个配置接口WebSocketConfigurer,我们来实现它,并启用@EnableWebSocket

import org.springframework.web.socket.config.annotation.EnableWebSocket;
import org.springframework.web.socket.config.annotation.WebSocketConfigurer;
​
@Configuration
@EnableWebSocket
public class WebSocketConfig implements WebSocketConfigurer {
​
 /**
 * 用于将WebSocket处理程序映射到特定URL
 *
 * @param registry
 */
 @Override
 public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
 registry.addHandler(chatWebSocketHandler(), "chat") // 添加消息处理器
 .setAllowedOrigins("*"); // 设置跨域
 }

 /**
 * 自定义消息处理器
 *
 * @return
 */
 @Bean
 public ChatWebSocketHandler chatWebSocketHandler() {
 return new ChatWebSocketHandler();
 }
​
}

重写registerWebSocketHandlers方法,通过registry.addHandler()将消息处理器添加,并指定映射的path,服务端WebSocket地址为 ws://host:port/path

注:需要注意的一点,这里要设置跨域,使用setAllowedOrigins方法即可。

至此WebSocket的服务端就可以使用了,还有一些其他骚操作继续往下看

WebSocket握手拦截器

在建立连接前,我们可以通过握手拦截器来拦截非法请求,需要实现SpringHandshakeInterceptor接口

import org.springframework.web.socket.server.HandshakeInterceptor;
​
public class ChatHandshakeInterceptor implements HandshakeInterceptor {
​
 private final Logger log = LoggerFactory.getLogger(this.getClass());
​
 @Override
 public boolean beforeHandshake(ServerHttpRequest request, ServerHttpResponse response, WebSocketHandler wsHandler, Map<String, Object> attributes) throws Exception {
 log.info("--------------握手前拦截");
 return true;
 }
​
 @Override
 public void afterHandshake(ServerHttpRequest request, ServerHttpResponse response, WebSocketHandler wsHandler, Exception exception) {
 log.info("--------------完成握手");
 }
}

实现两个方法:

  • beforeHandshake握手前,该方法返回true表示继续建立连接,返回false则终止

  • afterHandshake握手后

WebSocketConfig配置文件增加

@Override
public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
 registry.addHandler(chatWebSocketHandler(), "chat") // 添加消息处理器
 .addInterceptors(chatHandshakeInterceptor()) // 添加握手拦截器
 .setAllowedOrigins("*"); // 设置跨域
}
/**
 * 自定义消息拦截器
 *
 * @return
 */
@Bean
public ChatHandshakeInterceptor chatHandshakeInterceptor() {
 return new ChatHandshakeInterceptor();
}
Session空闲失效时间配置

一个会话连接后不可能一直不断开,这需要增加一些配置来约束超时时间;还可以设置消息缓冲区大小等(如果Nginx中配置了超时时间,此处可以忽略),上代码

WebSocketConfig配置文件增加

/**
 * 其他配置,如session空闲失效时间,消息缓冲区大小
 *
 * @return
 */
@Bean
public ServletServerContainerFactoryBean createWebSocketContainer() {
 ServletServerContainerFactoryBean container = new ServletServerContainerFactoryBean();
 container.setMaxTextMessageBufferSize(8192);
 container.setMaxSessionIdleTimeout(10 * 60 * 1000L);
 return container;
}
Nginx配置

大部分服务器都使用了Nginx代理,那么需要增加一些配置来支持WS协议

# 支持WS协议
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "upgrade";
​
# 会话超时时间
proxy_connect_timeout 620;
proxy_send_timeout 620;
proxy_read_timeout 620;

Client端

浏览器通过 JavaScript 向服务器发出建立 WebSocket 连接的请求,连接建立以后,客户端和服务器端就可以通过 TCP 连接直接交换数据。

创建WebSocket连接

连接协议使用wspath为服务端消息处理器映射的path

var ws = new WebSocket("ws://127.0.0.1:8080/chat");
WebSocket事件与方法
if (typeof (WebSocket) == "undefined") {
    alert("您的浏览器不支持WebSocket")
    return
}

// 打开一个 web socket
var ws = new WebSocket("ws://127.0.0.1:8080/chat");

ws.onopen = function()
{
    // Web Socket 已连接上,使用 send() 方法发送数据
    ws.send("发送数据");
    alert("数据发送中...");
};

ws.onmessage = function (evt) 
{ 
    var received_msg = evt.data;
    alert("数据已接收...");
};

ws.onclose = function()
{ 
    // 关闭 websocket
    alert("连接已关闭..."); 
};

ws.onerror = function () {
    // 连接错误
    alert("连接错误..."); 
    ws.close();
}

四个监听事件

  • onopen 与服务端连接成功

  • onmessage 接收服务端发送的消息

  • onclose 连接关闭

  • onerror 连接异常

两个方法

  • send() 向服务端发送消息

  • close()关闭与服务端的连接

聊天Demo演示

本demo简单实现用户A向用户B发送消息,用户B也可以回复消息,具体内容请移步我的码云!

demo.png

完整代码

springboot-websocket

timg (2).jpg
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,444评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,421评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,363评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,460评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,502评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,511评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,280评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,736评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,014评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,190评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,848评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,531评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,159评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,411评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,067评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,078评论 2 352