金歌 发表于 2023-10-11 16:29:32

SpringBoot进阶教程(七十七)WebSocket

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

很多网站为了实现推送技术,所用的技术都是轮询。轮询是在特定的时间间隔(如每1秒),由浏览器对服务器发出HTTP请求,然后由服务器返回最新的数据给客户端的浏览器。这种传统的模式带来很明显的缺点,即浏览器需要不断的向服务器发出请求,然而HTTP请求可能包含较长的头部,其中真正有效的数据可能只是很小的一部分,显然这样会浪费很多的带宽等资源。
而比较新的技术去做轮询的效果是Comet。这种技术虽然可以双向通信,但依然需要反复发出请求。而且在Comet中,普遍采用的长链接,也会消耗服务器资源。
在这种情况下,HTML5定义了WebSocket协议,能更好的节省服务器资源和带宽,并且能够更实时地进行通讯。
v架构搭建

添加maven引用
      <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-websocket</artifactId>
      </dependency>
      <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-thymeleaf</artifactId>
      </dependency>配置应用属性
server.port=8300
spring.thymeleaf.mode=HTML
spring.thymeleaf.cache=true
spring.thymeleaf.prefix=classpath:/web/
spring.thymeleaf.encoding: UTF-8
spring.thymeleaf.suffix: .html
spring.thymeleaf.check-template-location: true
spring.thymeleaf.template-resolver-order: 1添加WebSocketConfig
package com.test.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.socket.config.annotation.EnableWebSocket;
import org.springframework.web.socket.server.standard.ServerEndpointExporter;

/**
* @Author chen bo
* @Date 2023/10
* @Des
*/
@Configuration
public class WebSocketConfig {
    /**
   * bean注册:会自动扫描带有@ServerEndpoint注解声明的Websocket Endpoint(端点),注册成为Websocket bean。
   * 要注意,如果项目使用外置的servlet容器,而不是直接使用springboot内置容器的话,就不要注入ServerEndpointExporter,因为它将由容器自己提供和管理。
   */
    @Bean
    public ServerEndpointExporter serverEndpointExporter() {
      return new ServerEndpointExporter();
    }
}添加WebSocket核心类
因为WebSocket是类似客户端服务端的形式(采用ws协议),那么这里的WebSocketServer其实就相当于一个ws协议的Controller
直接@ServerEndpoint("/imserver/{userId}") 、@Component启用即可,然后在里面实现@OnOpen开启连接,@onClose关闭连接,@onMessage接收消息等方法。
新建一个ConcurrentHashMap用于接收当前userId的WebSocket或者Session信息,方便IM之间对userId进行推送消息。单机版实现到这里就可以。集群版(多个ws节点)还需要借助 MySQL或者 Redis等进行订阅广播方式处理,改造对应的 sendMessage方法即可。
package com.test.util;

import com.google.gson.JsonParser;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;

import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
import com.google.gson.JsonObject;

/**
* WebSocket的操作类
* html页面与之关联的接口
* var reqUrl = "http://localhost:8300/websocket/" + cid;
* socket = new WebSocket(reqUrl.replace("http", "ws"));
*/
@Component
@Slf4j
@ServerEndpoint("/websocket/{sid}")
public class WebSocketServer {

    /**
   * 静态变量,用来记录当前在线连接数,线程安全的类。
   */
    private static AtomicInteger onlineSessionClientCount = new AtomicInteger(0);

    /**
   * 存放所有在线的客户端
   */
    private static Map<String, Session> onlineSessionClientMap = new ConcurrentHashMap<>();

    /**
   * 连接sid和连接会话
   */
    private String sid;
    private Session session;

    /**
   * 连接建立成功调用的方法。由前端new WebSocket触发
   *
   * @param sid   每次页面建立连接时传入到服务端的id,比如用户id等。可以自定义。
   * @param session 与某个客户端的连接会话,需要通过它来给客户端发送消息
   */
    @OnOpen
    public void onOpen(@PathParam("sid") String sid, Session session) {
      /**
         * session.getId():当前session会话会自动生成一个id,从0开始累加的。
         */
      log.info("连接建立中 ==> session_id = {}, sid = {}", session.getId(), sid);
      //加入 Map中。将页面的sid和session绑定或者session.getId()与session
      //onlineSessionIdClientMap.put(session.getId(), session);
      onlineSessionClientMap.put(sid, session);

      //在线数加1
      onlineSessionClientCount.incrementAndGet();
      this.sid = sid;
      this.session = session;
      sendToOne(sid, "上线了");
      log.info("连接建立成功,当前在线数为:{} ==> 开始监听新连接:session_id = {}, sid = {},。", onlineSessionClientCount, session.getId(), sid);
    }

    /**
   * 连接关闭调用的方法。由前端socket.close()触发
   *
   * @param sid
   * @param session
   */
    @OnClose
    public void onClose(@PathParam("sid") String sid, Session session) {
      //onlineSessionIdClientMap.remove(session.getId());
      // 从 Map中移除
      onlineSessionClientMap.remove(sid);

      //在线数减1
      onlineSessionClientCount.decrementAndGet();
      log.info("连接关闭成功,当前在线数为:{} ==> 关闭该连接信息:session_id = {}, sid = {},。", onlineSessionClientCount, session.getId(), sid);
    }

    /**
   * 收到客户端消息后调用的方法。由前端socket.send触发
   * * 当服务端执行toSession.getAsyncRemote().sendText(xxx)后,前端的socket.onmessage得到监听。
   *
   * @param message
   * @param session
   */
    @OnMessage
    public void onMessage(String message, Session session) {
      /**
         * html界面传递来得数据格式,可以自定义.
         * {"sid":"user","message":"hello websocket"}
         */
      JsonObject jsonObject = JsonParser.parseString(message).getAsJsonObject();
      String toSid = jsonObject.get("sid").getAsString();
      String msg = jsonObject.get("message").getAsString();
      log.info("服务端收到客户端消息 ==> fromSid = {}, toSid = {}, message = {}", sid, toSid, message);

      /**
         * 模拟约定:如果未指定sid信息,则群发,否则就单独发送
         */
      if (toSid == null || toSid == "" || "".equalsIgnoreCase(toSid)) {
            sendToAll(msg);
      } else {
            sendToOne(toSid, msg);
      }
    }

    /**
   * 发生错误调用的方法
   *
   * @param session
   * @param error
   */
    @OnError
    public void onError(Session session, Throwable error) {
      log.error("WebSocket发生错误,错误信息为:" + error.getMessage());
      error.printStackTrace();
    }

    /**
   * 群发消息
   *
   * @param message 消息
   */
    private void sendToAll(String message) {
      // 遍历在线map集合
      onlineSessionClientMap.forEach((onlineSid, toSession) -> {
            // 排除掉自己
            if (!sid.equalsIgnoreCase(onlineSid)) {
                log.info("服务端给客户端群发消息 ==> sid = {}, toSid = {}, message = {}", sid, onlineSid, message);
                toSession.getAsyncRemote().sendText(message);
            }
      });
    }

    /**
   * 指定发送消息
   *
   * @param toSid
   * @param message
   */
    private void sendToOne(String toSid, String message) {
      // 通过sid查询map中是否存在
      Session toSession = onlineSessionClientMap.get(toSid);
      if (toSession == null) {
            log.error("服务端给客户端发送消息 ==> toSid = {} 不存在, message = {}", toSid, message);
            return;
      }
      // 异步发送
      log.info("服务端给客户端发送消息 ==> toSid = {}, message = {}", toSid, message);
      toSession.getAsyncRemote().sendText(message);
      /*
      // 同步发送
      try {
            toSession.getBasicRemote().sendText(message);
      } catch (IOException e) {
            log.error("发送消息失败,WebSocket IO异常");
            e.printStackTrace();
      }*/
    }

}添加controller
package com.test.controller;

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletResponse;

/**
* @Author chen bo
* @Date 2023/10
* @Des
*/
@Controller
public class HomeController {
    /**
   * 跳转到websocketDemo.html页面,携带自定义的cid信息。
   * http://localhost:8300/demo/toWebSocketDemo/user
   *
   * @param cid
   * @param model
   * @return
   */
    @GetMapping("/demo/toWebSocketDemo/{cid}")
    public String toWebSocketDemo(@PathVariable String cid, Model model) {
      model.addAttribute("cid", cid);
      return "index";
    }

    @GetMapping("hello")
    @ResponseBody
    public String hi(HttpServletResponse response) {
      return "Hi";
    }
}添加html
注意:html文件添加在application.properties配置的对应目录中。
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>聊天窗口</title>
</head>
<body>

我的用户名:
<input type="text" th:value="${cid}" readonly="readonly" id="cid"/>


收消息人用户名:<input id="toUserId" name="toUserId" type="text">
输入你要说的话:<input id="contentText" name="contentText" type="text">

    <button type="button" onclick="sendMessage()">发送消息</button>

</body>


</html>1对1模拟演练
启动项目后,在浏览器访问http://localhost:8300/demo/toWebSocketDemo/{cid} 跳转到对应页面,其中cid是用户名。
为了便于1对1测试,这里我们启动两个浏览器窗口。
http://localhost:8300/demo/toWebSocketDemo/阳光男孩
http://localhost:8300/demo/toWebSocketDemo/水晶女孩
按照要求输入对方用户信息之后,便可以输入你要说的话,畅快聊起来了。
效果图如下:
https://img2023.cnblogs.com/blog/506684/202309/506684-20230927193034783-1740980652.png
当然,如果收消息人用户名是自己的话,也可以自己给自己发送数据的。
群发模拟演练
为了便于群发测试,这里我们启动3个浏览器窗口。
http://localhost:8300/demo/toWebSocketDemo/阳光男孩
http://localhost:8300/demo/toWebSocketDemo/水晶女孩
http://localhost:8300/demo/toWebSocketDemo/路人A
由于sendToAll方法中定义群发的条件为:当不指定 toUserid时,则为群发。
效果图如下:
https://img2023.cnblogs.com/blog/506684/202309/506684-20230927193044844-863334369.png
项目架构图如下:
https://img2023.cnblogs.com/blog/506684/202309/506684-20230927211103637-1773280608.png
v源码地址

https://github.com/toutouge/javademosecond
其他参考/学习资料:

[*]https://www.cnblogs.com/xswz/p/10314351.html
[*]https://www.cnblogs.com/xuwenjin/p/12664650.html
[*]https://blog.csdn.net/qq_42402854/article/details/130948270
[*]https://www.cnblogs.com/zhangxinhua/p/11341292.html

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
页: [1]
查看完整版本: SpringBoot进阶教程(七十七)WebSocket