骏域网站建设专家广州,施工企业会计核算及常用会计分录,做网站公司松江,网站建设的市场情况在SSM项目中配置websocket
最近在ssm项目中配置了websocket#xff0c;踩了很多坑#xff0c;来分享一下
本文暂不提供发送消息等内容的代码逻辑#xff08;后续也许会补充#xff09;#xff0c;如果你直接复制这类可能会对配置造成更大的麻烦#xff08;博主就是复制…在SSM项目中配置websocket
最近在ssm项目中配置了websocket踩了很多坑来分享一下
本文暂不提供发送消息等内容的代码逻辑后续也许会补充如果你直接复制这类可能会对配置造成更大的麻烦博主就是复制了他人逻辑导致连接后秒断开连接排查了很长时间建议看完本文配置成功后去其他文章或者github寻找相应代码逻辑
1.Maven依赖
首先是jackson的依赖 dependencygroupIdcom.fasterxml.jackson.core/groupIdartifactIdjackson-annotations/artifactIdversion2.3.0/version/dependencydependencygroupIdcom.fasterxml.jackson.core/groupIdartifactIdjackson-core/artifactIdversion2.3.1/version/dependencydependencygroupIdcom.fasterxml.jackson.core/groupIdartifactIdjackson-databind/artifactIdversion2.3.3/version/dependency然后是spring关于websocket的依赖 dependencygroupIdorg.springframework/groupIdartifactIdspring-websocket/artifactIdversion${spring.version}/version/dependencydependencygroupIdorg.springframework/groupIdartifactIdspring-messaging/artifactIdversion${spring.version}/version/dependency这里的${spring.version}需要注意一定要和自己的spring版本匹配所以推荐用这种形式而不是直接用x.x.x.RELEASE
另外在spring4.0.5.RELEASE之前与其以后session获取参数的方法有些不同需要注意
本文使用的是
spring.version4.2.5.RELEASE/spring.version注意请删除所有关于javax的依赖将本地的tomcat引入库中否则可能会出现类型转换异常导致连接报错你可以继续下面的步骤如果在启动测试时出现该问题再回来尝试解决
2.配置类
首先在我们的项目中创建如下结构
当然也可以不使用这样的结构如果你的项目中本来有config、Interceptor、handler包可以将这些类放入其中
注意以下代码没有给出包名如需复制请注意
首先是WebSocketConfig.java
import org.springframework.stereotype.Component;
import org.springframework.web.socket.config.annotation.EnableWebSocket;
import org.springframework.web.socket.config.annotation.WebSocketConfigurer;
import org.springframework.web.socket.config.annotation.WebSocketHandlerRegistry;import javax.annotation.Resource;/*** WebScoket配置处理器*/
Component
EnableWebSocket
public class WebSocketConfig implements WebSocketConfigurer {ResourceMyWebSocketHandler handler;ResourceHandShake handShake;public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {//浏览器支持websocketregistry.addHandler(handler, /ws).addInterceptors(handShake);//浏览器不支持的话需要socketjs引入支持registry.addHandler(handler, /ws/sockjs).addInterceptors(handShake).withSockJS();}}这个类的作用是配置websocket的链接地址分为浏览器支持websocket和不支持websocket的情况不支持可以引入sockjs某些环境下使用这行代码会导致无法连接不放心可以删去
websocket连接地址是
ws://ip:端口号/项目名称/wsws与上述配置类相对应
这个连接地址是可以以?的形式传参数的例如
ws://ip:端口号/项目名称/ws?userIduserId第二个类是HandShake.java
import org.springframework.http.server.ServerHttpRequest;
import org.springframework.http.server.ServerHttpResponse;
import org.springframework.http.server.ServletServerHttpRequest;
import org.springframework.stereotype.Component;
import org.springframework.web.socket.WebSocketHandler;
import org.springframework.web.socket.server.HandshakeInterceptor;import javax.servlet.http.HttpSession;
import java.util.Map;/*** Socket建立连接握手和断开*/
Component
public class HandShake implements HandshakeInterceptor {Overridepublic boolean beforeHandshake(ServerHttpRequest request, ServerHttpResponse response, WebSocketHandler wsHandler, MapString, Object attributes) throws Exception {// 1.获取sessionServletServerHttpRequest servletRequest (ServletServerHttpRequest) request;HttpSession session servletRequest.getServletRequest().getSession(false);// 2.从session中获取uidLong uid (Long) session.getAttribute(uid);// 3.做正向判断如果uid是空值直接返回false拒绝握手if (uid null){return false;}// 4.输出日志System.out.println(Websocket:用户[ID: uid ]准备进行握手);// 5.将用户id存入mapattributes.put(uid, uid);// 6.可以进行握手return true;}Overridepublic void afterHandshake(ServerHttpRequest request, ServerHttpResponse response, WebSocketHandler wsHandler, Exception exception) {System.out.println(after hand);}}这个类是websocket的握手拦截器会拦截websocket链接地址的请求进行握手不建议在该类中做业务处理
最后是MyWebSocketHandler.java
import org.springframework.stereotype.Component;
import org.springframework.web.socket.*;
import java.io.IOException;/*** Socket处理器*/
Component
public class MyWebSocketHandler implements WebSocketHandler {Overridepublic void afterConnectionEstablished(WebSocketSession session)throws Exception {System.out.println(ConnectionEstablished);}Overridepublic void handleMessage(WebSocketSession session, WebSocketMessage? message) throws Exception {}/*** 消息传输错误处理*/Overridepublic void handleTransportError(WebSocketSession session,Throwable exception) throws Exception {System.out.println(TransportError);}/*** 关闭连接后*/Overridepublic void afterConnectionClosed(WebSocketSession session, CloseStatus closeStatus) throws Exception {System.out.println(ConnectionClosed);}Overridepublic boolean supportsPartialMessages() {return false;}
}这个类是对websocket的各种状态作出相应的处理如开头所说本文不会给出相应的逻辑防止出现错误这里真的很坑
大体思路就是通过方法中的WebSocketSession获取到session中的用户id再注入自己的userService或者其他service层去完成业务同时可以在本类中使用ConcurrentHashMap或者直接开一个线程来实现同时在线人数等功能
发送信息的逻辑推荐对信息进行封装构建实体类
此外你还可以创建一个相应的Controller通过注入MyWebSocketHandler来建立相应功能的请求地址
import com.webchat.websocket.MyWebSocketHandler;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;RestController
RequestMapping(/websocket)
public class WebSocketController {AutowiredMyWebSocketHandler handler;}3.JS部分
最后是js部分 let websocket new WebSocket(ws://ip:端口号/项目名称/ws?userIduserId);websocket.onmessage function(event) {console.log(event.data)};websocket.onopen function (event){console.log(连接成功)}websocket.onerror function (event){console.log(连接错误)}websocket.onclose function (event){console.log(websocket 断开: event.code event.reason event.wasClean)}注意更改地址
4.测试结果
测试结果如下 至此ssm项目中配置websocket就完成了
如果在测试中出现其他错误可以根据输出的event.code来匹配错误原因如果连接后直接断开并显示1011请检查后端代码规范性
编码原因1000正常关闭 当你的会话成功完成时发送这个代码1001离开 因应用程序离开且不期望后续的连接尝试而关闭连接时发送这一代码。服务器可能关闭或者客户端应用程序可能关闭1002协议错误 当因协议错误而关闭连接时发送这一代码1003不可接受的数据类型 当应用程序接收到一条无法处理的意外类型消息时发送这一代码1004保留 不要发送这一代码。根据 RFC 6455这个状态码保留可能在未来定义1005保留 不要发送这一代码。WebSocket API 用这个代码表示没有接收到任何代码1006保留 不要发送这一代码。WebSocket API 用这个代码表示连接异常关闭1007无效数据 在接收一个格式与消息类型不匹配的消息之后发送这一代码。如果文本消息包含错误格式的 UTF-8 数据连接应该用这个代码关闭1008消息违反政策 当应用程序由于其他代码所不包含的原因终止连接或者不希望泄露消息无法处理的原因时发送这一代码1009消息过大 当接收的消息太大应用程序无法处理时发送这一代码记住帧的载荷长度最多为64 字节。即使你有一个大服务器有些消息也仍然太大。1010需要扩展 当应用程序需要一个或者多个服务器无法协商的特殊扩展时从客户端浏览器发送这一代码1011意外情况 当应用程序由于不可预见的原因无法继续处理连接时发送这一代码1015TLS失败保留 不要发送这个代码。WebSocket API 用这个代码表示 TLS 在 WebSocket 握手之前失败。0 999禁止 1000 以下的代码是无效的不能用于任何目的1000 2999保留 这些代码保留以用于扩展和 WebSocket 协议的修订版本。按照标准规定使用这些代码参见表 3-43000 3999需要注册 这些代码用于“程序库、框架和应用程序”。这些代码应该在 IANA互联网编号分配机构公开注册4000 4999私有 在应用程序中将这些代码用于自定义用途。因为它们没有注册所以不要期望它们能被其他 WebSocket广泛理解