websocket实现web消息推送(基于Java注解实现websocket服务终端)

一、什么是web消息推送

Web消息通知是指Web系统向用户发送的有关系统更新、重要事件或消息的通知,以便用户及时了解系统的最新动态。在App和网页应用中,消息通知是最常见的信息交换方式之一,用于产品与用户之间的信息同步,如产品更新、信息提醒、互动提醒、新消息通知等,从而为用户提供帮助,使其能够快速获取对应的通知信息。
如图所示就是一个文本消息推送的案例,当某一个登录用户收到新消息时,网页端能够实时更新消息的数量。
在这里插入图片描述
web消息推送在csdn官网的表现形式是,当有新消息到来时,消息处出现一个小红点,提示有未读消息。
在这里插入图片描述

二、web消息推送的实现方式

web消息推送的实现方式有很多种,包括短轮询、长轮询、SSE、iframe流、websocket、mqtt等。本文主要主要讲解通过websockt的方式实现web消息推送。

三、websocket 实现消息推送

1.服务端

1.引入依赖

     <!--websocket依赖包-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-websocket</artifactId>
        </dependency>

2.websocketConfig配置类

package com.etime.websocket;

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

/**
 * @Description websocket 配置类
 * @Date 2024/5/18 20:28
 * @Author liukang
 **/
@Configuration
public class WebSocketConfig {
    @Bean
    public ServerEndpointExporter serverendpointExporter(){
        return new ServerEndpointExporter();
    }
}

3.实体类

package com.etime.entity;

import lombok.Data;

/**
 * @Date 2024/5/24 14:10
 * @Author liukang
 **/
@Data
public class MessagePo {
    private String msg;
    private String sendUser;
    private String toUser;
}

4.websockt终端类

package com.etime.websocket;

import com.etime.entity.MessagePo;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.extern.slf4j.Slf4j;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;

import javax.websocket.*;
import javax.websocket.server.PathParam;
import javax.websocket.server.ServerEndpoint;
import java.io.IOException;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;

/**
 * @Description 监听websocket地址  /myWs
 * @Date 2024/5/18 20:10
 * @Author liukang
 **/
@ServerEndpoint("/myWs/{userId}")
@Component
@Slf4j
public class WsServerEndPoint {
    // ConcurrentHashMap 和hashMap的区别是 ConcurrentHashMap是线程安全的
    static Map<String,Session> sessionMap = new ConcurrentHashMap<>();
    /**
     *  websocket建立链接时触发方法
     * @author liukang
     * @date 20:15 2024/5/18
     * @param session 每一个websocket的链接 对于服务端都是一个session
     **/
    @OnOpen
    public void onOpen(Session session, @PathParam("userId") String userId){
        // 建立连接时,将session存入map中 使用用户ID作为sessionMap的key
      sessionMap.put(userId,session);
      log.info("websocket is open");
      log.info(userId);

    }

    /**
     * 收到了客户端消息时触发方法
     * @author liukang
     * @date 20:20 2024/5/18
     * @param text 客户端传来的消息内容
     * @return java.lang.String
     **/
    @OnMessage
    public String onMessage(String text){
        return null;
    }
    /**
     * 连接关闭时触发方法
     * @author liukang
     * @date 20:21 2024/5/18
     * @param session 每一个websocket的链接 对于服务端都是一个session
     **/
    @OnClose
    public void onClose(Session session, @PathParam("userId") String userId){
        log.info(userId);
        sessionMap.remove(userId);
        log.info(("websocket is close"));
    }

 
	 /**
     * 发送消息
     * @author liukang
     * @date 15:49 2024/5/24 
     * @param message 
     **/
    public void sendMsgToUser(MessagePo message) throws IOException {
        String toUserId = message.getToUser();
        Session session = sessionMap.get(toUserId);
        RemoteEndpoint.Basic basicRemote = session.getBasicRemote();
        ObjectMapper mapper = new ObjectMapper();
        String msgStr = mapper.writeValueAsString(message);
        basicRemote.sendText(msgStr);
    }

}

关键点:
每个客户端的用户连接websocket服务终端的时候,携带用户唯一id(用户名唯一的话,也可以使用用户名),在建立websocket连接时,将连接保存到sessionMap,其中把用户userId作为sessionMap的key。这样做的目的是,如果要给某个用户发送消息时,根据用户id就能获取到服务终端与该用户的session连接,这样就可以对该用户发送消息了。
在这里插入图片描述
5.controller

package com.etime.controller;

import com.etime.entity.MessagePo;
import com.etime.websocket.WsServerEndPoint;
import org.springframework.web.bind.annotation.CrossOrigin;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import javax.annotation.Resource;
import java.io.IOException;

/**
 * @Date 2024/5/24 14:08
 * @Author liukang
 **/
@RestController
public class MessageController {
    @Resource
    private WsServerEndPoint wsServerEndPoint;// 引入websocket终端
    @RequestMapping("/sendMsg")
    @CrossOrigin  // 解决跨域问题
    public void sendMsg(@RequestBody MessagePo message) throws IOException {
        // 1.将消息数据插入到数据库--- 此处演示省略这步
        // 2.websocket发送通知
        wsServerEndPoint.sendMsgToUser(message);
    }
}


在MessageController 类中接受前端的发送消息请求;并引入websocket终端类,掉用该类中发送消息的方法进行消息websocket发送。

6.启动类

package com.etime;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;


/**
 * @Author liukang
 * @Date 2022/7/4 11:32
 */
@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class);
    }
}

2.客户端

1.写一个假的登录页面,模拟登录。
在这里插入图片描述
2.打开多个浏览器窗口,登录多个用户。
窗口1:
在这里插入图片描述
窗口2:
在这里插入图片描述
3.登录成功后连接websocket,关键代码如下

		 // 连接websocket
        let ws = new WebSocket("ws://localhost:8090/myWs/"+this.user.userId)
        ws.onopen = function(){
          console.log('open')
        
        }
      ws.onmessage = function(msg){
	        // 当前用户收到消息时,前端在这里进行处理,将消息处设置小红点或者数字;具体处理方式根据业务需求来
          console.log("接收到的消息:"+msg.data)
          console.log('接收到消息后消息处增加小红点进行提示')
        
        }
        ws.onclose = function(){
          console.log('close')
        }

注意点:连接websockt服务终端时,需要携带用户id
在这里插入图片描述

4.发送消息关键代码如下

  this.axios.post('http://localhost:8090/sendMsg',{
          
            sendUser:this.user.userId,
            toUser:this.toUserId,
            msg:this.content,
          
        })  
          .then(response => {  
            console.log(response.data);  
          })  
          .catch(error => {  
            console.error(error);  
          }); 
      }

3.效果演示

admin1用户向admin2用户发送消息,如下图
在这里插入图片描述
admin2窗口效果如下
在这里插入图片描述
为了方便,这里前端并没有进行具体的实现,只在控制台进行了打印。


  • 30
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot是一个开源的Java开发框架,它提供了许多对开发者友好的功能和工具。其中,Spring Boot还支持WebSocket,使得开发者能够轻松地实现消息推送功能。 在Spring Boot中使用WebSocket实现消息推送,需要以下几个步骤: 1. 添加WebSocket依赖:在项目的pom.xml文件中添加Spring Boot WebSocket依赖。 2. 创建WebSocket配置类:新建一个WebSocketConfig.java类,并在该类中添加@Configuration和@EnableWebSocket注解,以启用WebSocket。 3. 实现WebSocketHandle类:新建一个WebSocketHandler类,并继承TextWebSocketHandler类或实现WebSocketHandler接口,以处理WebSocket相关的请求和消息。在该类中,可以覆盖父类或接口的方法,如handleTextMessage()方法用于处理文本消息。 4. 添加WebSocket处理器:在WebSocketConfig.java类中,重写registerWebSocketHandlers()方法,并添加WebSocket处理器(即上一步创建的WebSocketHandler类)及对应的处理路径。 5. 前端页面:在前端页面中,使用WebSocket API与后端进行通信,并接收和处理后端发送的消息。 通过以上步骤,就完成了Spring Boot WebSocket的基本配置和实现。在实际使用中,可以根据具体需求,添加一些其他功能,如身份认证、消息广播、点对点消息等。 总结起来,Spring Boot通过提供WebSocket的支持,使得开发者能够方便地实现消息推送功能。通过简单的配置和编写处理器类,就可以实现前端的实时通信。这样,开发者可以轻松地在项目中应用消息推送的功能,实现更好的用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值