SpringBoot 整合 WebSocket 服务代码教程

return null;

}

//进行双重校验,两参数非空,且token中解析的用户id应于header传递的一致,否则视为非法请求

//websocket此处不再进行登录功能,均采用jwttoken进行身份验证

BladeUser bu = AuthUtil.getUser(token);

if (Objects.isNull(bu) || !StringUtil.equals(String.valueOf(bu.getUserId()), userId)) {

return null;

}

log.info(“用户:” + bu.getRealName() + “; userId:” + bu.getUserId() + " 连接成功");

accessor.setUser(() -> userId);

}

return message;

}

});

}

/**

  • 方法描述: 出界通道拦截配置,此处用来进行日志信息收集

  • @param registration

  • @Return

  • @author caichengzhe

  • @date 2021年07月08日 10:44:06

*/

@Override

public void configureClientOutboundChannel(ChannelRegistration registration) {

registration.interceptors(new ChannelInterceptor() {

@Override

public void postSend(Message<?> message, MessageChannel channel, boolean sent) {

log.info(“发出消息:” + JSON.toJSONString(message.getPayload()) + “;发送结果:” + (sent ? “成功” : “失败”) + “;消息通道:” + channel);

}

});

}

@Override

public void configureMessageBroker(MessageBrokerRegistry registry) {

//total message mapping include broadcast and specific topic

registry.enableSimpleBroker(“/topic/broadcast”, “/queue”);

// client-to-server message mapping prefix

registry.setApplicationDestinationPrefixes(“/unify-ws”);

// server to specific client mapping prefix

registry.setUserDestinationPrefix(“/user”);

}

}

或者 简单配置

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;

import org.springframework.web.socket.server.standard.ServerEndpointExporter;

@Configuration

public class WebSocketConfig {

/**

  • 注入一个ServerEndpointExporter,该Bean会自动注册使用@ServerEndpoint注解申明的websocket endpoint

*/

@Bean

public ServerEndpointExporter serverEndpointExporter() {

return new ServerEndpointExporter();

}

}

WebSocketService 服务类

import io.swagger.annotations.Api;

import lombok.extern.slf4j.Slf4j;

import org.springframework.stereotype.Component;

import javax.websocket.*;

import javax.websocket.server.ServerEndpoint;

import java.io.IOException;

import java.util.concurrent.ConcurrentHashMap;

import java.util.concurrent.atomic.AtomicInteger;

@Slf4j

@ServerEndpoint(value = “/point/change”)

@Component

@Api(value = “点位实时数据变更”, tags = “点位实时数据变更”)

public class WebSocketService {

/** concurrent包的线程安全Set,用来存放每个客户端对应的WebSocketServer对象。*/

private static ConcurrentHashMap<String, WebSocketService> webSocketMap = new ConcurrentHashMap<>();

private static ConcurrentHashMap<String, String> messageMap = new ConcurrentHashMap<>();

/*与某个客户端的连接会话,需要通过它来给客户端发送数据/

private Session session;

/** 记录当前在线连接数 */

private static AtomicInteger onlineCount = new AtomicInteger(0);

/**

  • 连接建立成功调用的方法

*/

@OnOpen

public void onOpen(Session session) {

this.session = session;

String sessionId = session.getId();

if(webSocketMap.containsKey(sessionId)){

//加入set中

webSocketMap.remove(sessionId);

webSocketMap.put(sessionId,this);

}else{

//加入set中

webSocketMap.put(sessionId,this);

// 在线数加1

onlineCount.incrementAndGet();

}

log.info(“有新连接加入:{},当前在线人数为:{}”, session.getId(), onlineCount.get());

}

/**

  • 连接关闭调用的方法

*/

@OnClose

public void onClose() {

if(webSocketMap.containsKey(session.getId())){

webSocketMap.remove(session.getId());

//从set中删除

onlineCount.decrementAndGet(); // 在线数减1

}

log.info(“有一连接关闭:{},当前在线人数为:{}”, session.getId(), onlineCount.get());

}

/**

  • 收到客户端消息后调用的方法

  • @param message

  •        客户端发送过来的消息
    

*/

@OnMessage

public void onMessage(String message, Session session) {

messageMap.remove(session.getId());

messageMap.put(session.getId(),message);

log.info(“服务端收到客户端[{}]的消息:{}”, session.getId(), message);

// this.sendMessage("Hello, " + message, session);

}

@OnError

public void onError(Session session, Throwable error) {

log.error(“发生错误”);

error.printStackTrace();

}

/**

  • 服务端发送消息给客户端

*/

public void sendMessage(String message, Session toSession) {

try {

log.info(“服务端给客户端[{}]发送消息{}”, toSession.getId(), message);

toSession.getBasicRemote().sendText(message);

} catch (Exception e) {

log.error(“服务端发送消息给客户端失败:{}”, e);

}

}

/**

  • 实现服务器主动推送

*/

public void sendMessage(String message) throws IOException {

synchronized (session) {

this.session.getBasicRemote().sendText(message);

}

}

Java面试核心知识点笔记

其中囊括了JVM、锁、并发、Java反射、Spring原理、微服务、Zookeeper、数据库、数据结构等大量知识点。

蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6

Java中高级面试高频考点整理

蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6

蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6

最后分享Java进阶学习及面试必备的视频教学

蚂蚁金服(Java研发岗),26岁小伙斩获三面,收获Offer定级P6

ion.getBasicRemote().sendText(message);

}

}

Java面试核心知识点笔记

其中囊括了JVM、锁、并发、Java反射、Spring原理、微服务、Zookeeper、数据库、数据结构等大量知识点。

[外链图片转存中…(img-R5XOJN4L-1714459771382)]

Java中高级面试高频考点整理

[外链图片转存中…(img-gnfg6tJ0-1714459771383)]

[外链图片转存中…(img-GZHgBoKM-1714459771383)]

最后分享Java进阶学习及面试必备的视频教学

[外链图片转存中…(img-Po2IIIln-1714459771384)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

  • 10
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值