Java Websocket 01: 原生模式 Websocket 基礎通信

来源:https://www.cnblogs.com/milton/archive/2023/06/18/17489013.html
-Advertisement-
Play Games

原生模式下, 服務端通過 @ServerEndpoint 實現其對應的 @OnOpen, @OnClose, @OnMessage, @OnError 方法, 客戶端創建 WebSocketClient 實現對應的 onOpen(), onClose(), onMessage(), onError(... ...


目錄

Websocket 原生模式

原生模式下

  • 服務端通過 @ServerEndpoint 實現其對應的 @OnOpen, @OnClose, @OnMessage, @OnError 方法
  • 客戶端創建 WebSocketClient 實現對應的 onOpen(), onClose(), onMessage(), onError()

演示項目

完整示例代碼 https://github.com/MiltonLai/websocket-demos/tree/main/ws-demo01

目錄結構

│   pom.xml
└───src
    ├───main
    │   ├───java
    │   │   └───com
    │   │       └───rockbb
    │   │           └───test
    │   │               └───wsdemo
    │   │                       SocketServer.java
    │   │                       WebSocketConfig.java
    │   │                       WsDemo01App.java
    │   └───resources
    │           application.yml
    └───test
        └───java
            └───com
                └───rockbb
                    └───test
                        └───wsdemo
                                SocketClient.java

pom.xml

  • 可以用 JDK11, 也可以用 JDK17
  • 通過 Spring Boot plugin repackage, 生成 fat jar
  • 用 Java-WebSocket 作為 client 的 websocket 實現庫, 當前最新版本為 1.5.3
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.rockbb.test</groupId>
    <artifactId>ws-demo01</artifactId>
    <packaging>jar</packaging>
    <version>1.0-SNAPSHOT</version>

    <name>WS: Demo 01</name>

    <properties>
        <!-- Global encoding -->
        <project.jdk.version>17</project.jdk.version>
        <project.source.encoding>UTF-8</project.source.encoding>
        <!-- Global dependency versions -->
        <spring-boot.version>2.7.11</spring-boot.version>
    </properties>

    <dependencyManagement>
        <dependencies>
            <!-- Spring Boot Dependencies -->
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-dependencies</artifactId>
                <version>${spring-boot.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-websocket</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-messaging</artifactId>
        </dependency>

        <dependency>
            <groupId>org.java-websocket</groupId>
            <artifactId>Java-WebSocket</artifactId>
            <version>1.5.3</version>
        </dependency>

    </dependencies>

    <build>
        <finalName>ws-demo01</finalName>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.10.1</version>
                <configuration>
                    <source>${project.jdk.version}</source>
                    <target>${project.jdk.version}</target>
                    <encoding>${project.source.encoding}</encoding>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-resources-plugin</artifactId>
                <version>3.3.0</version>
                <configuration>
                    <encoding>${project.source.encoding}</encoding>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <version>${spring-boot.version}</version>
                <executions>
                    <execution>
                        <goals>
                            <goal>repackage</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>
</project>

application.yml

設置服務埠為 8763

server:
  port: 8763
  tomcat:
    uri-encoding: UTF-8

spring:
  application:
    name: ws-demo01

WsDemo01App.java

  • 將 @RestController 也合併到應用入口了. 和單獨拆開做一個 Controller 類是一樣的
  • '/msg' 路徑用於從 server 往 client 發送消息
@RestController
@SpringBootApplication
public class WsDemo01App {

    public static void main(String[] args) {
        SpringApplication.run(WsDemo01App.class, args);
    }

    @RequestMapping("/msg")
    public String sendMsg(String sessionId, String msg) throws IOException {
        Session session = SocketServer.getSession(sessionId);
        SocketServer.sendMessage(session, msg);
        return "send " + sessionId + " : " + msg;
    }
}

WebSocketConfig.java

必須顯式聲明 ServerEndpointExporter 這個 Bean 才能提供 websocket 服務

@Configuration
public class WebSocketConfig {

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

SocketServer.java

提供 websocket 服務的關鍵類. @ServerEndpoint 的作用類似於 RestController, 這裡指定 client 訪問的路徑格式為 ws://host:port/websocket/server/[id],
當 client 訪問使用不同的 id 時, 會對應產生不同的 SocketServer 實例

@Component
@ServerEndpoint("/websocket/server/{sessionId}")
public class SocketServer {
    private static final org.slf4j.Logger log = LoggerFactory.getLogger(SocketServer.class);
    private static final Map<String, Session> sessionMap = new ConcurrentHashMap<>();

    private String sessionId = "";

    @OnOpen
    public void onOpen(Session session, @PathParam("sessionId") String sessionId) {
        this.sessionId = sessionId;
        /* Old connection will be kicked by new connection */
        sessionMap.put(sessionId, session);
        /*
         * this: instance id. New instances will be created for each sessionId
         * sessionId: assigned from path variable
         * session.getId(): the actual session id (start from 0)
         */
        log.info("On open: this{} sessionId {}, actual {}", this, sessionId, session.getId());
    }

    @OnClose
    public void onClose() {
        sessionMap.remove(sessionId);
        log.info("On close: sessionId {}", sessionId);
    }

    @OnMessage
    public void onMessage(String message, Session session) {
        log.info("On message: sessionId {}, {}", session.getId(), message);
    }

    @OnError
    public void onError(Session session, Throwable error) {
        log.error("On error: sessionId {}, {}", session.getId(), error.getMessage());
    }

    public static void sendMessage(Session session, String message) throws IOException {
        session.getBasicRemote().sendText(message);
    }

    public static Session getSession(String sessionId){
        return sessionMap.get(sessionId);
    }
}

關於會話對象 Session

OnOpen 會註入一個 Session 參數, 這個是實際的 Websocket Session, 其 ID 是全局唯一的, 可以唯一確定一個客戶端連接. 在當前版本的實現中, 這是一個從0開始自增的整數. 如果你需要實現例如單個用戶登錄多個會話, 在通信中, 將消息轉發給同一個用戶的多個會話, 就要小心記錄這些 Session 的 ID.

@OnOpen
public void onOpen(Session session, @PathParam("sessionId") String sessionId)

關於會話意外關閉

在客戶端意外停止後, 服務端會收到 OnError 消息, 可以通過這個消息管理已經關閉的會話

SocketClient.java

client 測試類, 連接後可以通過命令行向 server 發送消息

public class SocketClient {

    private static final org.slf4j.Logger log = LoggerFactory.getLogger(SocketClient.class);

    public static void main(String[] args) throws URISyntaxException {

        WebSocketClient wsClient = new WebSocketClient(
                new URI("ws://127.0.0.1:8763/websocket/server/10001")) {

            @Override
            public void onOpen(ServerHandshake serverHandshake) {
                log.info("On open: {}, {}", serverHandshake.getHttpStatus(), serverHandshake.getHttpStatusMessage());
            }

            @Override
            public void onMessage(String s) {
                log.info("On message: {}", s);
            }

            @Override
            public void onClose(int i, String s, boolean b) {
                log.info("On close: {}, {}, {}", i, s, b);
            }

            @Override
            public void onError(Exception e) {
                log.info("On error: {}", e.getMessage());
            }
        };

        wsClient.connect();
        log.info("Connecting...");
        while (!ReadyState.OPEN.equals(wsClient.getReadyState())) {
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                log.error(e.getMessage(), e);
            }
        }
        log.info("Connected");

        wsClient.send("hello");

        Scanner scanner = new Scanner(System.in);
        while (scanner.hasNext()) {
            String line = scanner.next();
            wsClient.send(line);
        }
        wsClient.close();
    }
}

代碼的執行過程就是新建一個 WebSocketClient 並實現其處理消息的介面方法, 使用 10001 作為 sessionId 進行連接, 在連接成功後, 不斷讀取鍵盤輸入 (System.in), 將輸入的字元串發送給服務端.

運行示例

示例是一個普通的 Spring Boot jar項目, 可以通過mvn clean package進行編譯, 再通過java -jar ws-demo01.jar運行, 啟動後工作在8763埠

然後運行 SocketClient.java, 可以觀察到服務端接收到的消息.

服務端可以通過瀏覽器訪問 http://127.0.0.1:8763/msg?sessionId=10001&msg=123 向客戶端發送消息.

結論

以上說明並演示了原生的 Websocket 實現方式, 可以嘗試運行多個 SocketClient, 使用相同或不同的 server sessionId 路徑, 觀察通信的變化情況.


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 近日,極限數據 (北京) 科技有限公司(以下簡稱:極限科技)旗下的軟體 INFINI Easysearch 搜索引擎軟體 V1.0 通過統信 UOS 伺服器操作系統 V20 認證。 此次相容適配基於統信 UOS 伺服器操作系統 V20,聯合國產 CPU:海光 5000、海光 7000、兆芯 KH-3 ...
  • [TOC](MySQL主鍵、唯一索引、聯合索引的區別和作用) # 0. 簡介 索引是一類特殊的`文件`,用來存儲檢索信息,使資料庫查找更加快速。 # 1. 主鍵 主鍵是一類特殊的唯一索引,選擇某一列元素作為主鍵,用來表示每一行元素的特殊性,其特點如下 - 在一個數據表中只有一個主鍵; - 主鍵不能為 ...
  • 本文主要基於團隊實際開發經驗與積累,並結合了業界對大數據SQL的使用與優化,嘗試給出相對系統性的解決方案。 ...
  • ## 1 安裝環境 ### Node.js js的運行環境,相當於 java 的 jvm 官網:https://nodejs.org/en,下載最新穩定版 `18.16.0 LTS`,雙擊安裝即可 自動安裝了npm,終端驗證: ```bash C:\Users\Administrator>node ...
  • # React SSR - 寫個 Demo 一學就會 今天寫個小 `Demo` 來從頭實現一下 `react` 的 `SSR`,幫助理解 `SSR` 是如何實現的,有什麼細節。 ## 什麼是 SSR `SSR` 即 `Server Side Rendering` 服務端渲染,是指將網頁內容在伺服器端 ...
  • ## 介紹 這是一款基於VUE3.0 打造的簡約型博客主題,相容各大主流瀏覽器,適配各個設備與解析度,PC、平板、手機等均可正常瀏覽。並且採用響應式設計,提高使用響應速度。 ## 特性 - 響應式設計,相容平板、手機端瀏覽器。 - 提供多種配置信息,方便各類用戶進行個人定製化。 - 部署文檔十分詳細 ...
  • ### 歡迎訪問我的GitHub > 這裡分類和彙總了欣宸的全部原創(含配套源碼):[https://github.com/zq2599/blog_demos](https://github.com/zq2599/blog_demos) ### 本篇概覽 - 本文是《JavaCV的攝像頭實戰》的第八 ...
  • > 2023/6/18 > > 本篇章記錄學習過程C++的基礎概念和代碼測試實現,還有很多需要補充。一是還不清楚,二是還沒有學到。打算學習過程中後面再做補充。先看完《C++primer 》書之後再慢慢來添加補充 # 1.函數重載 1. 一個函數名可以實現多個功能,這取決於函數參數不同來實現判斷對應的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...