基于websocket实现im聊天

2023-11-14 14:53:25 浏览数 (1)

前言

文字聊天应该是很多人每天常用的功能,这篇文章就来分析一下聊天是怎么创建的,他的底层逻辑是什么,以及如何实现他的底层逻辑。

设计实现

消息的发送过程就像这张图一样,经过websocket进行一个消息的转发,一对一是这样,在群里里面发消息也是这个逻辑,一个消息发送给多个人。本质是也是一对一聊天,只是逻辑的干预是你觉得他就是一对多群聊。

用户缓存的核心代码,缓存用户连接。SessionCache封装了用户的信息和客户端连接的引用。

代码语言:javascript复制
/**
 * 保存sessionId与用户的map
 */
Map<String, SessionCache> sessionCacheMap = new ConcurrentHashMap<>();

当用户登录的时候建立websocket长连接并且把连接信息存入到本地缓存,当有用户给他发消息的时候进行一个消息转发。

代码语言:javascript复制
public void wsOpen(WebSocketSession session) throws IOException {
    CacheToken cache = validateTokenAndReturnCache(session, tokenCache);
    if (cache == null) return;
    log.info("websocket id:{} 建立连接,用户token:{} 连接成功,放入在线用户缓存", session.getId(), cache.getToken());
    javsessionCacheMap.put(session.getId(), new SessionCache(cache.getToken(), session));
}

当用户下线之后的代码,从本地清除缓存,如果有用户给他发消息就存入到缓存。

代码语言:javascript复制
/**
 * 关闭连接清除缓存
* @param sessionId
* @return
 */
public void wsClose(String sessionId) {
	SessionCache sessionCache = sessionCacheMap.remove( sessionId );
}

发送消息的代码,当消息过来获取缓存中的WebSocketSession进行一个字节流的逻辑写入。

代码语言:javascript复制
/**
 * 发送消息的方法
* @param session
* @param payload
* @return
 */
public boolean sendData(WebSocketSession session, String payload){
	if (session == null || !session.isOpen()) {
return false;
}
	try {
		if (session.isOpen()) {
			session.sendMessage(new TextMessage(payload));
			return true;
		} else {
		// 连接关闭,丢弃消息
		// String token = (String) session.getAttributes().get("token");
 		// log.debug( "websocket id:{} 已经关闭, token:{}, 丢弃消息。", session.getId(), token);
 		// log.debug("websocket关闭丢弃消息:{}", payload);
			return false;
		}
	} catch (IOException e) {
		if (e instanceof IOException || e instanceof SocketTimeoutException) {
			log.info("发送消息报错:session.isOpen():{} ERROR: {}", session != null ? session.isOpen() : null,
			e.getMessage());
		} else {
			log.error("发送消息报错:session.isOpen():{}", session != null ? session.isOpen() : null, e);
		}
		return false;
	}
}

消息体

常见的消息分为文字消息,图片消息还有语音消息。这里暂且不说视频流。也就是我们常用的视频聊天。首先从最基本的文字,图片和语音来说,到后台这边都是使用的JSON格式。文字就不用说了,图片的话是经过一个第三方的存储文件服务器转换成的一个链接。然后用户端进行一个渲染。然后才会看到常用的一些图片。然后语音消息其实也是一段文凭。音频其实也是一个文件。到文件服务器之后也是一个文件的链接,然后用户端进去。格式进行一个转换。最后呈现给用户的就是一段语音。

这里要说的有一个点,就是一个敏感字的处理。其实敏感字的检索这就涉及到一些算法的操作。我们可能需要查看某一段文字当中是不是有一些敏感字?在常见的操作当然是使用字符检索一下,也可以使用一些专门的检索算法,当然逻辑上每一个系统里面会有一张专门用来配置的敏感字目录,存在敏感字。进行拒绝或者打码操作。

总结

这里解释了最简单的语音聊天的一个。实现当你用户量变得很大的时候,可能已经满足不了需求。需要用到Netty进行一个性能的提升。这是后话大道至简,其实越复杂的东西是靠着简单的功能一点点往上面叠加而产生的,一对一的聊天,本质上就是数据的传递。

引用

https://www.baeldung.com/java-websockets

点赞关注评论一键三连,欢迎关注公众号【i查拉图斯特拉如是说】每周分享技术干货、开源项目、实战经验、国外优质文章翻译等,您的关注将是我的更新动力!

我正在参与2023腾讯技术创作特训营第三期有奖征文,组队打卡瓜分大奖!

0 人点赞