@ServerEndpoint注解:2023最新分享,springboot中轻量级websocket

@ServerEndpoint:

主要是将目前的类定义成一个websocket服务器端, 注解的值将被用于监听用户连接的终端访问URL地址,客户端可以通过这个URL来连接到WebSocket服务器端

说明:本项目是springboot集成websocket
我项目用的是gradel引入依赖,下边附上maven的依赖,version与springboot保持一致即可

build.gradle
    compile group:  'org.springframework.boot', name: 'spring-boot-starter-websocket', version: '2.0.4.RELEASE'

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

本文采用注解式编程,快速开发如下:
核心处理器 MyWebSocketServer.java


@ServerEndpoint(value = "/mysocket", encoders = AnswerEncoder.class)
@Component
@Slf4j
public class MyWebSocketServer {

    /**
     * 静态变量,用来记录当前在线连接数。应该把它设计成线程安全的。
     */
    private static int onlineCount = 0;

    /**
     * concurrent包的线程安全Set,用来存放每个客户端对应的MyWebSocket对象。
     */
    private static final CopyOnWriteArraySet<ScreenWebSocketServer> WEB_SOCKET_SET = new CopyOnWriteArraySet<ScreenWebSocketServer>();

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

    /**
     * 连接建立成功调用的方法
     */
    @OnOpen
    public void onOpen(Session session) {
        this.session = session;
        // 将当前连接加入到set中
        WEB_SOCKET_SET.add(this);
        // 连接数+1
        addOnlineCount();
        log.info("MyWebSocketServer 加入新的连接,当前连接数为" + getOnlineCount());
    }

    /**
     * 连接关闭调用的方法
     */
    @OnClose
    public void onClose() {
        // 将当前连接从set中移除
        WEB_SOCKET_SET.remove(this);
        // 连接数-1
        subOnlineCount();
        log.info("MyWebSocketServer 有一连接关闭!当前连接数为" + getOnlineCount());
    }

    /**
     * 收到客户端消息后调用的方法
     *
     * @param message 客户端发送过来的消息
     */
    @OnMessage
    public void onMessage(String message, Session session) {
        log.info("MyWebSocketServer 收到来自窗口" + session.getId() + "的信息:" + message);
    }

    /**
     * @param session 连接session
     * @param error   措施信息
     */
    @OnError
    public void onError(Session session, Throwable error) {
        log.error("MyWebSocketServer 连接发生错误");
        error.printStackTrace();
    }


    /**
     * 

@Description:給session连接推送消息

*

@param [message]

*

@return void

*

@throws

*/
private void sendMessage(Object message) throws IOException { try { this.session.getBasicRemote().sendObject(message); } catch (EncodeException e) { e.printStackTrace(); log.error("MyWebSocketServer 向客户端推送数据发生错误"); } } /** *

@Description:向所有连接群发消息

*

@param [message]

*

@return void

*

@throws

*/
public static void sendMessageToAll(Object message){ for (ScreenWebSocketServer item : WEB_SOCKET_SET) { try { item.sendMessage(message); } catch (IOException e) { log.error("MyWebSocketServer 向客户端推送数据发生错误"); } } } public static synchronized int getOnlineCount() { return onlineCount; } public static synchronized void addOnlineCount() { ScreenWebSocketServer.onlineCount++; } public static synchronized void subOnlineCount() { ScreenWebSocketServer.onlineCount--; } @Override public boolean equals(Object o) { if (this == o) { return true; } if (o == null || getClass() != o.getClass()) { return false; } ScreenWebSocketServer that = (ScreenWebSocketServer) o; return Objects.equals(session, that.session); } @Override public int hashCode() { return Objects.hash(session); } }

发送消息的编码器 AnswerEncoder .java

public class AnswerEncoder implements Encoder.Text<Answer> {
    @Override
    public void destroy() {
        // TODO Auto-generated method stub
    }

    @Override
    public void init(EndpointConfig arg0) {
        // TODO Auto-generated method stub
    }

    @Override
    public String encode(Answer answer) throws EncodeException {
        return JSONUtil.toJsonStr(answer);
    }
}

最最最关键的一步:将ServerEndpointExporter 暴露给spring容器管理,它会帮我们注入标注了@ServerEndpoint注解的websocket处理器,否则上面配置的不生效。

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

/**
 * File Name: WebsocketConfiguration
 * Description: Websocket 相关配置类
 */
@Configuration
public class WebSocketConfiguration {

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

}

到此为止我们的websocket就配置好了,那么如何调用呢,如何从服务端主动推送message给客户端呢?请看如下实例代码:


@RequestMapping("socketPushTest")
@RestController
public class MyWebSocketPushTest {

    @ApiOperation("给已连接的所有session进行群发")
    @PostMapping("/sendMessageToAll")
    public void push(@RequestBody Object data) {
        // 给前端群发数据变更消息
        Answer success = Answer.success("群发的消息-----啦啦啦");
        MyWebSocketServer.sendMessageToAll(success);
    }
}

核心代码就是 MyWebSocketServer.sendMessageToAll(success);
好了,现在我们的springboot内嵌的websocket就配置好了,我们可以通过url的方式连接到这个服务

下面贴一下我测试的图
说明:我的springboot项目启动端口(也就是yml文件的server.port)是9090
ws://127.0.0.1:9090/mysocket
步骤1.模拟ws请求用户1
@ServerEndpoint注解:2023最新分享,springboot中轻量级websocket_第1张图片
模拟ws请求用户2
@ServerEndpoint注解:2023最新分享,springboot中轻量级websocket_第2张图片
查看idea控制台打印日志
@ServerEndpoint注解:2023最新分享,springboot中轻量级websocket_第3张图片
此时ws连接都创建成功,调用ws推送接口,实现群发,然后查看用户1和用户2是否能收到
@ServerEndpoint注解:2023最新分享,springboot中轻量级websocket_第4张图片
@ServerEndpoint注解:2023最新分享,springboot中轻量级websocket_第5张图片

问题答疑:配置后,springboot项目启动报错
本博客使用范围是springboot+tomcat的,如果启动报错请在maven或者gradle依赖引入的时候排除
spring-boot-starter-tomcat,配置如下:

build.gradle
compile("org.springframework.boot:spring-boot-starter-web:2.1.4.RELEASE") {
        exclude module: "spring-boot-starter-tomcat"
        exclude group: 'org.springframework.boot', module: 'spring-boot-starter-logging'
    }
pom.xml
	<dependency>
		<groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-websocket</artifactId>
		<exclusions>
			<exclusion>
				<groupId>org.springframework.boot</groupId>
				<artifactId>tomcat-embed-websocket</artifactId>
			</exclusion>
		</exclusions>
	</dependency>


如果您的项目发布服务器是jetty,可以出门左转了,本博客不适用。本博客所写代码是将websocket用tomcat发布,与本身的springboot是一体的,不额外启用端口!!!!

你可能感兴趣的:(websocket,spring,boot,websocket,java)