当前位置: 首页 > article >正文

Golang WebSocket 创建单独会话

引言

在互联网应用程序中,实时通信是一种非常重要的功能。WebSocket 是一种基于 TCP 的协议,它允许客户端和服务器之间进行双向通信。Golang 是一种高性能的编程语言,它提供了对 WebSocket 的原生支持,使得在 Golang 中创建 WebSocket 会话变得非常简单。本文将介绍如何使用 Golang 创建单独的 WebSocket 会话,以实现实时通信功能。

WebSocket 简介

WebSocket 是一种在单个 TCP 连接上进行全双工通信的协议。它与传统的 HTTP 协议不同,HTTP 是一种无状态的协议,每个请求都需要建立一个新的连接。而 WebSocket 在客户端和服务器之间建立一条持久的连接,可以实现实时的双向通信。

WebSocket 协议的握手是通过 HTTP 请求完成的,握手后,客户端和服务器之间的连接将保持打开状态,可以发送和接收任意数据。WebSocket 使用一种类似于事件的机制,当有新消息到达时,服务器可以主动推送给客户端,而不需要客户端主动发送请求。

Golang 中的 WebSocket 支持

Golang 提供了 net/http 包来处理 HTTP 请求和响应,同时也提供了 gorilla/websocket 库来实现 WebSocket 协议的支持。gorilla/websocket 是一个非常流行的第三方库,它提供了对 WebSocket 协议的高级封装,使得在 Golang 中创建 WebSocket 会话变得更加简单。

在开始之前,我们首先需要安装 gorilla/websocket 库。可以使用以下命令来安装:

go get github.com/gorilla/websocket

安装完成后,我们就可以开始创建 WebSocket 会话了。

创建 WebSocket 服务器

首先,我们需要创建一个 WebSocket 服务器,用于接收来自客户端的连接请求,并处理 WebSocket 会话。以下是一个简单的 WebSocket 服务器示例:

package main

import (
    "log"
    "net/http"

    "github.com/gorilla/websocket"
)

// 创建一个 upgrader 对象,用于升级 HTTP 连接为 WebSocket 连接
var upgrader = websocket.Upgrader{
    ReadBufferSize:  1024,
    WriteBufferSize: 1024,
}

func main() {
    http.HandleFunc("/ws", handleWebSocket)
    log.Fatal(http.ListenAndServe(":8080", nil))
}

func handleWebSocket(w http.ResponseWriter, r *http.Request) {
    // 将 HTTP 连接升级为 WebSocket 连接
    conn, err := upgrader.Upgrade(w, r, nil)
    if err != nil {
        log.Println("Upgrade failed:", err)
        return
    }

    defer conn.Close()

    // 处理 WebSocket 会话
    for {
        // 读取客户端发送的消息
        messageType, message, err := conn.ReadMessage()
        if err != nil {
            log.Println("Read message failed:", err)
            break
        }

        // 处理客户端发送的消息
        log.Printf("Received message: %s", message)

        // 回复客户端消息
        err = conn.WriteMessage(messageType, message)
        if err != nil {
            log.Println("Write message failed:", err)
            break
        }
    }
}

在上面的示例中,我们首先创建了一个 upgrader 对象,它用于将 HTTP 连接升级为 WebSocket 连接。然后,我们定义了一个 handleWebSocket 函数,用于处理 WebSocket 会话。在该函数中,我们首先将 HTTP 连接升级为 WebSocket 连接,然后进入一个无限循环,不断读取客户端发送的消息,并给客户端回复相同的消息。

最后,我们使用 http.HandleFunc 函数将 /ws 路径映射到 handleWebSocket 函数,然后调用 http.ListenAndServe 函数来启动 WebSocket 服务器。

创建 WebSocket 客户端

接下来,我们需要创建一个 WebSocket 客户端,用于向服务器发送 WebSocket 请求,并处理服务器推送的消息。以下是一个简单的 WebSocket 客户端示例:

package main

import (
    "log"
    "os"
    "os/signal"
    "time"

    "github.com/gorilla/websocket"
)

func main() {
    // 创建一个 dialer 对象,用于建立 WebSocket 连接
    dialer := websocket.Dialer{
        HandshakeTimeout: 10 * time.Second,
    }

    // 建立 WebSocket 连接
    conn, _, err := dialer.Dial("ws://localhost:8080/ws", nil)
    if err != nil {
        log.Fatal("Dial failed:", err)
    }
    defer conn.Close()

    // 启动一个 goroutine 来接收服务器推送的消息
    go func() {
        for {
            _, message, err := conn.ReadMessage()
            if err != nil {
                log.Println("Read message failed:", err)
                break
            }

            log.Printf("Received message: %s", message)
        }
    }()

    // 向服务器发送消息
    message := []byte("Hello, WebSocket!")
    err = conn.WriteMessage(websocket.TextMessage, message)
    if err != nil {
        log.Println("Write message failed:", err)
        return
    }

    // 等待用户按下 Ctrl+C 终止程序
    interrupt := make(chan os.Signal, 1)
    signal.Notify(interrupt, os.Interrupt)
    <-interrupt
}

在上面的示例中,我们首先创建了一个 dialer 对象,它用于建立 WebSocket 连接。然后,我们使用 dialer.Dial 函数建立 WebSocket 连接,并指定服务器的地址为 ws://localhost:8080/ws。然后,我们使用 conn.WriteMessage 函数向服务器发送消息,并使用一个 goroutine 来接收服务器推送的消息。

最后,我们使用 signal.Notify 函数来注册一个信号监听器,当用户按下 Ctrl+C 时,程序会接收到一个中断信号,然后程序退出。

创建单独会话

在实际应用中,我们可能需要为每个客户端创建一个单独的会话,以便管理和跟踪客户端的状态。在 Golang 中,可以通过为每个 WebSocket 连接创建一个独立的 goroutine 来实现这一点。以下是一个示例:

package main

import (
    "log"
    "net/http"

    "github.com/gorilla/websocket"
)

var upgrader = websocket.Upgrader{
    ReadBufferSize:  1024,
    WriteBufferSize: 1024,
}

func main() {
    http.HandleFunc("/ws", handleWebSocket)
    log.Fatal(http.ListenAndServe(":8080", nil))
}

func handleWebSocket(w http.ResponseWriter, r *http.Request) {
    conn, err := upgrader.Upgrade(w, r, nil)
    if err != nil {
        log.Println("Upgrade failed:", err)
        return
    }

    defer conn.Close()

    // 创建一个独立的会话
    session := NewSession()

    // 处理会话
    session.Handle(conn)
}

type Session struct {
    conn *websocket.Conn
}

func NewSession() *Session {
    return &Session{}
}

func (s *Session) Handle(conn *websocket.Conn) {
    s.conn = conn

    go s.readLoop()
    go s.writeLoop()
}

func (s *Session) readLoop() {
    for {
        messageType, message, err := s.conn.ReadMessage()
        if err != nil {
            log.Println("Read message failed:", err)
            break
        }

        log.Printf("Received message: %s", message)
    }
}

func (s *Session) writeLoop() {
    for {
        select {
        // 从消息队列中获取消息并发送给客户端
        case message := <-s.messageQueue:
            err := s.conn.WriteMessage(websocket.TextMessage, message)
            if err != nil {
                log.Println("Write message failed:", err)
            }
        }
    }
}

在上面的示例中,我们首先定义了一个 Session 结构体,它包含一个 WebSocket 连接。然后,我们创建了一个 NewSession 函数,用于创建一个新的会话对象。会话对象有两个重要的方法:Handle 方法用

场景

Golang的WebSocket可以用于创建单独的会话,适用于许多场景。以下是一个使用场景的介绍:

场景:在线聊天室

在一个在线聊天室中,用户可以通过WebSocket与其他用户进行实时的文字交流。每个用户都可以创建一个单独的会话,与其他用户进行私聊或在群组中发送消息。

使用场景描述:

  1. 用户进入聊天室,并在前端页面上输入昵称和聊天内容。
  2. 前端页面通过WebSocket与后端的Golang服务器建立连接。
  3. 后端服务器使用Golang的WebSocket包处理客户端的连接请求。
  4. 当用户发送消息时,前端页面将消息通过WebSocket发送至后端服务器。
  5. 后端服务器接收到消息后,将其广播给所有在线的用户,或者根据需要仅发送给特定的用户。
  6. 每个连接的客户端都可以接收到其他用户发送的消息,并在前端页面上展示出来。

此场景中,Golang的WebSocket实现了用户之间的实时通信,并保持了每个用户的会话独立性。它可以处理并发连接,使得多个用户能够同时进行聊天,而不会相互干扰。

值得注意的是,Golang的WebSocket还可以通过添加必要的安全性和认证机制来确保聊天室的安全性。例如,可以使用SSL/TLS加密连接,或者使用令牌进行用户身份验证。这些安全性措施可以确保用户的聊天内容和个人信息得到保护。


http://www.kler.cn/news/108598.html

相关文章:

  • 【MySQL】并发事务产生的问题及事务隔离级别
  • 0028Java程序设计-智能农场监控报警系统设计与实现
  • 私有云:【8】VCenter安装Connection服务
  • Linux-sdio接口
  • 全志T113-S3 裸机SMHC eMMC读写问题记录
  • python:使用Scikit-image对遥感影像进行傅里叶变换特征提取(fourier)
  • C#使用mysql-connector-net驱动连接mariadb报错
  • 【MATLAB源码-第62期】基于蜣螂优化算法(DBO)的无人机三维地图路径规划,输出最短路径和适应度曲线。
  • Docker 启动远程服务访问不了
  • SpringMVC学习
  • Android 13 - Media框架(12)- MediaCodec(二)
  • 漏洞复现-dedecms文件上传(CVE-2019-8933)
  • AcWing第 127 场周赛 - AcWing 5283. 牛棚入住+AcWing 5284. 构造矩阵 - 模拟+快速幂+数学
  • SAP从入门到放弃系列之QM动态修改(Dynamic Modification)
  • 挖掘业务场景的存储更优解
  • Android开发知识学习——登录和第三方授权
  • 前端和后端 优化
  • javaEE -8(9000字详解网络编程)
  • 【面试经典150 | 栈】简化路径
  • 数字孪生与智慧城市:开启未来智慧生活
  • Spring Cloud之Docker的学习【详细】
  • 【每日一题】补档 CF487B. Strip | 数据结构杂烩 -> 单调队列 | 困难
  • 【python】pip的使用
  • C++之C++11引入enum class与传统enum关键字总结(二百五十一)
  • 如何通过adb控制安卓手机wifi
  • 手机apn介绍
  • 66 内网安全-域横向批量atschtasksimpacket
  • PostgreSQL InvalidMessage Cache 同步机制
  • 如何确保PCIe Gen3通道的信号质量
  • SpringBoot2.7.14整合redis7