• springboot 集成 WebSocket (非本人原创)


    WebSocket  应用场景:服务端向客户端发送数据。

    导入Maven

    <dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-websocket</artifactId>
    </dependency>

    创建config

    @Configuration
    public class WebSocketConfig {

    @Bean
    public ServerEndpointExporter serverEndpointExporter() {
    return new ServerEndpointExporter();
    }

    }

    WebSocketServer

    因为WebSocket是类似客户端服务端的形式(采用ws协议),那么这里的WebSocketServer其实就相当于一个ws协议的Controller 
    直接@ServerEndpoint(“/websocket”)@Component启用即可,然后在里面实现@OnOpen,@onClose,@onMessage等方法

    @ServerEndpoint("/websocket/{sid}")
    @Component
    public class WebSocketServer {

    static Log log=LogFactory.getLog(WebSocketServer.class);
    //静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
    private static int onlineCount = 0;
    //concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。
    private static CopyOnWriteArraySet<WebSocketServer> webSocketSet = new CopyOnWriteArraySet<WebSocketServer>();

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

    //接收sid
    private String sid="";
    /**
    * 连接建立成功调用的方法*/
    @OnOpen
    public void onOpen(Session session,@PathParam("sid") String sid) {
    this.session = session;
    webSocketSet.add(this); //加入set中
    addOnlineCount(); //在线数加1
    log.info("有新窗口开始监听:"+sid+",当前在线人数为" + getOnlineCount());
    this.sid=sid;
    try {
    sendMessage("连接成功");
    } catch (IOException e) {
    log.error("websocket IO异常");
    }
    }

    /**
    * 连接关闭调用的方法
    */
    @OnClose
    public void onClose() {
    webSocketSet.remove(this); //从set中删除
    subOnlineCount(); //在线数减1
    log.info("有一连接关闭!当前在线人数为" + getOnlineCount());
    }

    /**
    * 收到客户端消息后调用的方法
    *
    * @param message 客户端发送过来的消息*/
    @OnMessage
    public void onMessage(String message, Session session) {
    log.info("收到来自窗口"+sid+"的信息:"+message);
    //群发消息
    for (WebSocketServer item : webSocketSet) {
    try {
    item.sendMessage(message);
    } catch (IOException e) {
    e.printStackTrace();
    }
    }
    }

    /**
    *
    * @param session
    * @param error
    */
    @OnError
    public void onError(Session session, Throwable error) {
    log.error("发生错误");
    error.printStackTrace();
    }
    /**
    * 实现服务器主动推送
    */
    public void sendMessage(String message) throws IOException {
    this.session.getBasicRemote().sendText(message);
    }


    /**
    * 群发自定义消息
    * */
    public static void sendInfo(String message,@PathParam("sid") String sid) throws IOException {
    log.info("推送消息到窗口"+sid+",推送内容:"+message);
    for (WebSocketServer item : webSocketSet) {
    try {
    //这里可以设定只推送给这个sid的,为null则全部推送
    if(sid==null) {
    item.sendMessage(message);
    }else if(item.sid.equals(sid)){
    item.sendMessage(message);
    }
    } catch (IOException e) {
    continue;
    }
    }
    }

    public static synchronized int getOnlineCount() {
    return onlineCount;
    }

    public static synchronized void addOnlineCount() {
    WebSocketServer.onlineCount++;
    }

    public static synchronized void subOnlineCount() {
    WebSocketServer.onlineCount--;
    }
    }

    消息推送

    至于推送新信息,可以再自己的Controller写个方法调用WebSocketServer.sendInfo();即可

    @Controller @RequestMapping("/checkcenter") public class CheckCenterController { //页面请求 @GetMapping("/socket/{cid}") public ModelAndView socket(@PathVariable String cid) { ModelAndView mav=new ModelAndView("/socket"); mav.addObject("cid", cid); return mav; } //推送数据接口 @ResponseBody @RequestMapping("/socket/push/{cid}") public ApiReturnObject pushToWeb(@PathVariable String cid,String message) { try { WebSocketServer.sendInfo(message,cid); } catch (IOException e) { e.printStackTrace(); return ApiReturnUtil.error(cid+"#"+e.getMessage()); } return ApiReturnUtil.success(cid); } }

    页面发起socket请求

    然后在页面用js代码调用socket,当然,太古老的浏览器是不行的,一般新的浏览器或者谷歌浏览器是没问题的。还有一点,记得协议是ws的哦,如果像我这样封装了一些basePath的路径类,可以replace(“http”,”ws”)来替换协议

    <script> var socket; if(typeof(WebSocket) == "undefined") { console.log("您的浏览器不支持WebSocket"); }else{ console.log("您的浏览器支持WebSocket"); //实现化WebSocket对象,指定要连接的服务器地址与端口 建立连接 //等同于socket = new WebSocket("ws://localhost:8083/checkcentersys/websocket/20"); socket = new WebSocket("${basePath}websocket/${cid}".replace("http","ws")); //打开事件 socket.onopen = function() { console.log("Socket 已打开"); //socket.send("这是来自客户端的消息" + location.href + new Date()); }; //获得消息事件 socket.onmessage = function(msg) { console.log(msg.data); //发现消息进入 开始处理前端触发逻辑 }; //关闭事件 socket.onclose = function() { console.log("Socket已关闭"); }; //发生了错误事件 socket.onerror = function() { alert("Socket发生了错误"); //此时可以尝试刷新页面 } //离开页面时,关闭socket //jquery1.8中已经被废弃,3.0中已经移除 // $(window).unload(function(){ // socket.close(); //}); } </script>

    VUE 前端代码:

    send () {
    var message = 'haha'
    this.readyChat(message)
    },
    /* webSocket会话 */
    /* 会话过程中实际调用的函数 */
    readyChat (data) {
    let This = this
    if (this.websock.readyState === 1) { // this.websock.readyState = 1 表示连接成功,可以立即发送信息
    this.websocketSend(data)
    } else if (this.websock.readyState === 0) { // 表示正在连接,设置300ms后发送信息
    setTimeout(function () {
    This.websocketSend(data)
    }, 300)
    } else { // 连接未创建或者创建失败,则重新创建连接,并设置500ms后发送信息
    this.websochetInit()
    setTimeout(function () {
    This.websocketSend(data)
    }, 500)
    }
    },
    /* 初始化websochet */
    websochetInit () {
    this.websock = new WebSocket('ws:/192.168.0.。。。。')
    this.websock.onmessage = this.websocketMessage
    },
    /* websochet发送信息 */
    websocketSend (data) {
    console.log(data)
    this.websock.send(data)
    },
    /* websochet接收服务器返回的信息 */
    websocketMessage (e) {

    },
  • 相关阅读:
    debian/ubuntu系统vi无法删除字符的解决办法
    kvm磁盘镜像文件管理,格式转换,调整大小
    读懂MACD背离,多空力量分析
    OpenStack部署都有哪些方式
    [STM32F10x] 利用定时器测量频率
    [STM32F10x] 利用定时器测量脉冲宽度
    曼彻斯特编码
    最近关注的几个技术点网页链接
    windows中cmd常用命令收集
    Mybatis Generator代码自动生成(实体类、dao层、映射文件)
  • 原文地址:https://www.cnblogs.com/laixin09/p/9619413.html
Copyright © 2020-2023  润新知