当前位置:首页 > 文章 > 正文内容

websocket前后端数据实时更新(前端+后端)

廖万里3年前 (2022-10-27)文章124779

做的项目中需要将后端提供的数据展示在前端页面,一开始我是用JS的setInterval()方法,设置一个时间,每过时间发起一次ajax请求。虽然也能凑活着实现,但总感觉数据不是实时刷新的,而且还占用资源,所以学习WebSocke,并总结了一下,以下是本人总结的前后端WebSocke相关代码:


一、后端:

1.pom.xml添加WebSocke依赖

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

2.WebSocke配置类

@Configurationpublic class WebSocketConfig {    /**
     * 这个bean的注册,用于扫描带有@ServerEndpoint的注解成为websocket  ,如果你使用外置的tomcat就            
        不需要该配置文件
     */
    @Bean
    public ServerEndpointExporter serverEndpointExporter() {        return new ServerEndpointExporter();
    }

}

3.WebSocke服务类

@ServerEndpoint(value = "/webSocket")//主要是将目前的类定义成一个websocket服务器端, 注解的值将被用于监听用户连接的终端访问URL地址,客户端可以通过这个URL来连接到WebSocket服务器端@Component@EnableScheduling// cron定时任务@Datapublic class WebSocket {    private static final Logger logger = LoggerFactory.getLogger(WebSocket.class);    /**
     * 静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
     */
    private static int onlineCount = 0;    /**
     * concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。
     */
    private static CopyOnWriteArraySet<WebSocket> webSocketSet = new CopyOnWriteArraySet<>();    /**
     * 与某个客户端的连接会话,需要通过它来给客户端发送数据
     */
    private Session session;    public static CopyOnWriteArraySet<WebSocket> getWebSocketSet() {        return webSocketSet;
    }    public static void setWebSocketSet(CopyOnWriteArraySet<WebSocket> webSocketSet) {
        WebSocket.webSocketSet = webSocketSet;
    }    /**
     * 从数据库查询相关数据信息,可以根据实际业务场景进行修改
     */
    @Resource
    private IndexService indexService;    private static IndexService indexServiceMapper;    @PostConstruct
    public void init() {
        WebSocket.indexServiceMapper = this.indexService;
    }    /**
     * 连接建立成功调用的方法
     *
     * @param session 会话
     */
    @OnOpen
    public void onOpen(Session session) throws Exception {        this.session = session;
        webSocketSet.add(this);        //查询当前在线人数
        int nowOnline = indexServiceMapper.nowOnline();        this.sendMessage(JSON.toJSONString(nowOnline));
    }    /**
     * 收到客户端消息后调用的方法
     *
     * @param message 客户端发送过来的消息
     */
    @OnMessage
    public void onMessage(String message, Session session) throws IOException {
        logger.info("参数信息:{}", message);        //群发消息
        for (WebSocket item : webSocketSet) {            try {
                item.sendMessage(JSON.toJSONString(message));
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }    /**
     * 连接关闭调用的方法
     */
    @OnClose
    public void onClose() {
        webSocketSet.remove(this);        if (session != null) {            try {
                session.close();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }    /**
     * 发生错误时调用
     *
     * @param session 会话
     * @param error   错误信息
     */
    @OnError
    public void onError(Session session, Throwable error) {
        logger.error("连接异常!");
        error.printStackTrace();
    }    /**
     * 发送信息
     *
     * @param message 消息
     */
    public void sendMessage(String message) throws IOException {        this.session.getBasicRemote().sendText(message);
    }    /**
     * 自定义消息推送、可群发、单发
     *
     * @param message 消息
     */
    public static void sendInfo(String message) throws IOException {
        logger.info("信息:{}", message);        for (WebSocket item : webSocketSet) {
            item.sendMessage(message);
        }
    }
}

 4.定时任务(为了给前端实时推送数据,我这里写了个定时任务,定时任务我用的是cron表达式,不懂的同学可以上这个网址学习:cron表达式

@Slf4j@Componentpublic class IndexScheduled {    @Autowired
    private IndexMapper indexMapper;    /**
     * 每3秒执行一次
     */
    //@Scheduled(cron = "0/3 * * * * ? ") //我这里暂时不需要运行这条定时任务,所以将注解注释了,朋友们运行时记得放开注释啊
    public void nowOnline() {
        System.err.println("*********   首页定时任务执行   **************");

        CopyOnWriteArraySet<WebSocket> webSocketSet = WebSocket.getWebSocketSet();        int nowOnline = indexMapper.nowOnline();
        webSocketSet.forEach(c -> {            try {
                c.sendMessage(JSON.toJSONString(nowOnline));
            } catch (IOException e) {
                e.printStackTrace();
            }
        });

        System.err.println("/n 首页定时任务完成.......");
    }

}

二、前端:

前端的代码非常的简单,直接上代码。

<body class="gray-bg"><div class="online">
   <span class="online">测试在线人数:<span id="online"></span>&nbsp人</span></div><script th:inline="javascript">
    
    let websocket = null;    let host = document.location.host;    //判断当前浏览器是否支持WebSocket
    if ('WebSocket' in window) {        //连接WebSocket节点
        websocket = new WebSocket("ws://" + host + "/webSocket");
    } else {        alert('浏览器不支持webSocket');
    }    
    //连接发生错误的回调方法
    websocket.onerror = function () {        setMessageInnerHTML("error");
    };    
    //连接成功建立的回调方法
    websocket.onopen = function (event) {        setMessageInnerHTML("open");
    };    
    //接收到消息的回调方法
    websocket.onmessage = function (event) {        let data = event.data;        console.log("后端传递的数据:" + data);        //将后端传递的数据渲染至页面
        $("#online").html(data);
    };    
    //连接关闭的回调方法
    websocket.onclose = function () {        setMessageInnerHTML("close");
    };    
    //监听窗口关闭事件,当窗口关闭时,主动去关闭websocket连接,防止连接还没断开就关闭窗口,server端会抛异常。
    window.onbeforeunload = function () {
        websocket.close();
    };    
    //将消息显示在网页上
    function setMessageInnerHTML(innerHTML) {
    
    };    
    //关闭连接
    function closeWebSocket() {
        websocket.close();
    };    
    //发送消息
    function send() {        let message = document.getElementById('text').value;
        websocket.send(message);
    };    
</script></body>

 


本文链接:https://www.kkkliao.cn/?id=169 转载需授权!

分享到:

版权声明:本文由廖万里的博客发布,如需转载请注明出处。


“websocket前后端数据实时更新(前端+后端)” 的相关文章

不打游戏只看视频,骁龙和天玑竟然能拉开这么大差距?

不打游戏只看视频,骁龙和天玑竟然能拉开这么大差距?

事情是这样的。最近托尼有位同事因为之前被使用三星 4nm 工艺的骁龙 8 Gen 1 折腾怕了,所以他在把原来的旧手机卖了之后,转手换了台搭载天玑 9000 的手机。一开始他对这台手机可以说非常满意,打游戏时发热终于没那么严重了,然而时间一长,他发现手机电量貌似掉的有点快,续航并没有想象中那么顶。本...

四川公务员考试 真题_福建省选调生考试书籍

四川公务员考试 真题_福建省选调生考试书籍

而在四川的国家公务员考试中,行测中的言语理解题目甚至占到了40道题。所以,很多时候,你会听到“得言语者得行测”就是在强调言语理解在四川公务员考试行测中的重要...2019年四川省省考已进入备考阶段,刷历年省考真题可提升巩固管家整理了33套行测申论四川历练真题,带答案解析需要的同学,看下方图片领取方式...

骁龙8和骁龙8+的日常体验,到底有多大差距?

骁龙8和骁龙8+的日常体验,到底有多大差距?

机哥写过一篇文章。主要呢,就是盘点了今年那些跳水比较严重的旗舰手机。像什么OPPO Find X5 Pro天玑版啊、小米12 Pro啊、一加10 Pro啊等等。相比起刚上市时定价,现在这些机型,普遍的降价幅度都超过了2000块。原本是卖5000多,现在却只卖3000多。。机哥当时发完文章,底下很大一...

马云即便隐退了,眼光依旧毒辣,退出前的发言值得深思

马云即便隐退了,眼光依旧毒辣,退出前的发言值得深思

(ps:灰色的文字为马云的发言,黑色为作者的补充解读,更助于各位理解)马云发言:过去的这一年,很不寻常,事件发生了巨大的变化。疫情也带来了巨大的挑战,在今天所有巨大的不确定当中,有一件事是确定无疑的。那就是数字化的趋势没有改变。数字化以前只是让一些企业活得更好。而今天是企业活下去的关键,数字化的进程...

你最讨厌QQ什么?

你最讨厌QQ什么?

作为中国最早的社交软件之一, QQ承载着无数的青春。QQ最初诞生的时候,是为了方便人们之间的交流而诞生的。那时候 QQ作为我们联系的主要工具,人们之间可以进行即时通讯。我们每天都会在 QQ上和不同的人进行沟通互动。聊天的内容也十分的丰富,有的时候聊天的内容甚至超过了现在人们生活的内容。而其中最让人讨...

最简单的生活一天花多少钱?

最简单的生活一天花多少钱?

90后负债女孩的极简主义:月薪6000+,一天的真实花销精简但你绝对想不到!广西农村姑娘在广州,网贷负债6w,人情债接近3w,到手月薪6000+,在珠江新城商业CBD上班,一天真实的花销在多少?道出多少负债人的辛酸!疫情这3年的收入,固定死工资基本就这个数了,负债之下一直没啥存款,都是还没发工资都被...

发表评论

访客

看不清,换一张

◎欢迎参与讨论,请在这里发表您的看法和观点。