WebSocket 实现指南

news2025/1/8 19:08:49

WebSocket 实现指南

目录

1. 依赖安装

1.1 安装必要的包

# 安装 gorilla/websocket
go get github.com/gorilla/websocket

# 安装 gin 框架
go get github.com/gin-gonic/gin

1.2 更新 go.mod

require (
    github.com/gin-gonic/gin v1.9.1
    github.com/gorilla/websocket v1.5.3
)

1.3 配置文件

1.3.1Config.go
package config

import (
	"os"
	"gopkg.in/yaml.v3"
)

type Config struct {
	Server    ServerConfig    `yaml:"server"`
	MySQL     MySQLConfig     `yaml:"mysql"`
	Redis     RedisConfig     `yaml:"redis"`
	JWT       JWTConfig       `yaml:"jwt"`
	WebSocket WebSocketConfig `yaml:"websocket"`
}

type ServerConfig struct {
	Port int    `yaml:"port"`
	Mode string `yaml:"mode"`
}

type MySQLConfig struct {
	Host     string `yaml:"host"`
	Port     int    `yaml:"port"`
	User     string `yaml:"user"`
	Password string `yaml:"password"`
	DBName   string `yaml:"dbname"`
}

type RedisConfig struct {
	Host     string `yaml:"host"`
	Port     int    `yaml:"port"`
	Password string `yaml:"password"`
	DB       int    `yaml:"db"`
}

type JWTConfig struct {
	Secret string `yaml:"secret"`
	Expire int    `yaml:"expire"` // token过期时间(小时)
}

type WebSocketConfig struct {
	ReadBufferSize  int   `yaml:"read_buffer_size"`
	WriteBufferSize int   `yaml:"write_buffer_size"`
	MaxMessageSize  int64 `yaml:"max_message_size"`
	WriteWait       int   `yaml:"write_wait"`
	PongWait        int   `yaml:"pong_wait"`
	PingPeriod      int   `yaml:"ping_period"`
	MaxConnections  int   `yaml:"max_connections"`
}

var GlobalConfig Config

func Init() error {
	// 确保日志目录存在
	os.MkdirAll("logs", 0755)

	file, err := os.ReadFile("config/config.yaml")
	if err != nil {
		return err
	}

	return yaml.Unmarshal(file, &GlobalConfig)
}

1.3.2 config.yaml
server:
  port: 8080
  mode: debug  # debug or release

websocket:
  read_buffer_size: 1024
  write_buffer_size: 1024
  max_message_size: 512
  write_wait: 10     # seconds
  pong_wait: 60      # seconds
  ping_period: 54    # seconds
  max_connections: 5000

mysql:
  host: localhost
  port: 3306
  user: root
  password: "123456"
  dbname: my_kingdom

redis:
  host: localhost
  port: 6379
  password: "123456"
  db: 0

jwt:
  secret: "XueZhimin"
  expire: 24  # hours

2. 代码实现

2.1 WebSocket管理器 (manager.go)

package websocket

import (
	"encoding/json"
	"fmt"
	"mykingdom/config"
	"net/http"
	"sync"

	"github.com/gin-gonic/gin"
	"github.com/gorilla/websocket"
)

// Manager WebSocket管理器
type Manager struct {
	clients   sync.Map    // 所有客户端连接
	broadcast chan []byte // 广播消息通道
	config    *config.WebSocketConfig
	messages  chan Message // 新增:消息处理通道
}

// Message 定义消息结构
type Message struct {
	Type     string      `json:"type"`
	Data     interface{} `json:"data"`
	ClientID string      `json:"-"` // 发送者的连接ID
}

// 配置websocket upgrader
var upgrader = websocket.Upgrader{
	ReadBufferSize:  1024,
	WriteBufferSize: 1024,
	CheckOrigin: func(r *http.Request) bool {
		return true // 允许所有跨域请求
	},
}

// NewManager 创建WebSocket管理器
func NewManager(config *config.WebSocketConfig) *Manager {
	return &Manager{
		broadcast: make(chan []byte),
		messages:  make(chan Message, 256), // 新增:初始化消息通道
		config:    config,
	}
}

// HandleWebSocket 处理WebSocket连接
func (m *Manager) HandleWebSocket() gin.HandlerFunc {
	return func(c *gin.Context) {
		// 检查连接数限制
		count := 0
		m.clients.Range(func(_, _ interface{}) bool {
			count++
			return true
		})
		if count >= m.config.MaxConnections {
			c.JSON(http.StatusServiceUnavailable, gin.H{
				"message": "达到最大连接数限制",
			})
			return
		}

		conn, done := Upgrade(c)
		if done {
			return
		}

		// 创建新的客户端连接
		client := &Client{
			conn:    conn,
			manager: m,
			send:    make(chan []byte, 256),
		}

		// 存储客户端连接
		m.clients.Store(client.conn.RemoteAddr().String(), client)

		// 启动读写协程
		go client.readPump()
		go client.writePump()
	}
}

// Upgrade 升级HTTP连接为WebSocket连接
func Upgrade(c *gin.Context) (*websocket.Conn, bool) {
	conn, err := upgrader.Upgrade(c.Writer, c.Request, nil)
	if err != nil {
		return nil, true
	}
	return conn, false
}

// Broadcast 广播消息给所有客户端
func (m *Manager) Broadcast(message []byte) {
	m.clients.Range(func(_, value interface{}) bool {
		if client, ok := value.(*Client); ok {
			select {
			case client.send <- message:
			default:
				m.clients.Delete(client.conn.RemoteAddr().String())
				close(client.send)
			}
		}
		return true
	})
}

// SendToClient 发送消息给指定客户端
func (m *Manager) SendToClient(clientAddr string, message []byte) bool {
	if value, ok := m.clients.Load(clientAddr); ok {
		if client, ok := value.(*Client); ok {
			client.send <- message
			return true
		}
	}
	return false
}

// SendMessage 发送任意类型的消息
func (m *Manager) SendMessage(messageType string, data interface{}) error {
	message := Message{
		Type: messageType,
		Data: data,
	}

	// 将消息转换为JSON
	jsonData, err := json.Marshal(message)
	if err != nil {
		return fmt.Errorf("marshal message failed: %v", err)
	}

	// 广播消息
	m.Broadcast(jsonData)
	return nil
}

// SendMessageToClient 发送消息给指定客户端
func (m *Manager) SendMessageToClient(clientAddr string, messageType string, data interface{}) error {
	message := Message{
		Type: messageType,
		Data: data,
	}

	// 将消息转换为JSON
	jsonData, err := json.Marshal(message)
	if err != nil {
		return fmt.Errorf("marshal message failed: %v", err)
	}

	// 发送给指定客户端
	if !m.SendToClient(clientAddr, jsonData) {
		return fmt.Errorf("client not found: %s", clientAddr)
	}
	return nil
}

// GetMessages 获取消息通道,用于读取消息
func (m *Manager) GetMessages() <-chan Message {
	return m.messages
}

// ReadMessage 读取消息
func ReadMessage(conn *websocket.Conn) ([]byte, error) {
	_, message, err := conn.ReadMessage()
	if err != nil {
		return nil, fmt.Errorf("read message failed: %v", err)
	}
	return message, nil
}

// WriteMessage 发送消息
func WriteMessage(conn *websocket.Conn, message []byte) error {
	err := conn.WriteMessage(websocket.TextMessage, message)
	if err != nil {
		return fmt.Errorf("write message failed: %v", err)
	}
	return nil
}

// WriteJSON 发送JSON消息
func WriteJSON(conn *websocket.Conn, v interface{}) error {
	err := conn.WriteJSON(v)
	if err != nil {
		return fmt.Errorf("write json failed: %v", err)
	}
	return nil
}

2.2 客户端实现 (client.go)

package websocket

import (
	"encoding/json"
	"log"
	"time"

	"github.com/gorilla/websocket"
)

const (
	writeWait  = 10 * time.Second
	pongWait   = 60 * time.Second
	pingPeriod = (pongWait * 9) / 10
)

// Client WebSocket客户端
type Client struct {
	conn    *websocket.Conn
	manager *Manager
	send    chan []byte
}

// readPump 从WebSocket连接读取消息
func (c *Client) readPump() {
	defer func() {
		c.manager.clients.Delete(c.conn.RemoteAddr().String())
		c.conn.Close()
	}()

	c.conn.SetReadDeadline(time.Now().Add(pongWait))
	c.conn.SetPongHandler(func(string) error {
		c.conn.SetReadDeadline(time.Now().Add(pongWait))
		return nil
	})

	for {
		_, message, err := c.conn.ReadMessage()
		if err != nil {
			if websocket.IsUnexpectedCloseError(err, websocket.CloseGoingAway, websocket.CloseAbnormalClosure) {
				log.Printf("error: %v", err)
			}
			break
		}

		// 尝试解析消息为Message结构
		var msg Message
		if err := json.Unmarshal(message, &msg); err != nil {
			log.Printf("unmarshal message failed: %v", err)
			continue
		}

		// 设置发送者ID
		msg.ClientID = c.conn.RemoteAddr().String()

		// 将消息发送到消息通道
		c.manager.messages <- msg

		// 广播消息给所有客户端
		c.manager.Broadcast(message)
	}
}

// writePump 向WebSocket连接写入消息
func (c *Client) writePump() {
	ticker := time.NewTicker(pingPeriod)
	defer func() {
		ticker.Stop()
		c.conn.Close()
	}()

	for {
		select {
		case message, ok := <-c.send:
			c.conn.SetWriteDeadline(time.Now().Add(writeWait))
			if !ok {
				c.conn.WriteMessage(websocket.CloseMessage, []byte{})
				return
			}

			w, err := c.conn.NextWriter(websocket.TextMessage)
			if err != nil {
				return
			}
			w.Write(message)

			if err := w.Close(); err != nil {
				return
			}
		case <-ticker.C:
			c.conn.SetWriteDeadline(time.Now().Add(writeWait))
			if err := c.conn.WriteMessage(websocket.PingMessage, nil); err != nil {
				return
			}
		}
	}
}

3. 配置说明

3.1 WebSocket配置 (config.yaml)

websocket:
  read_buffer_size: 1024    # 读缓冲区大小
  write_buffer_size: 1024   # 写缓冲区大小
  max_connections: 5000     # 最大连接数

3.2 配置结构体 (config.go)

type WebSocketConfig struct {
    ReadBufferSize  int   `yaml:"read_buffer_size"`
    WriteBufferSize int   `yaml:"write_buffer_size"`
    MaxConnections  int   `yaml:"max_connections"`
}

4. 使用示例

4.1 服务端示例

func main() {
    r := gin.Default()
    
    // 创建WebSocket管理器
    wsManager := websocket.NewManager(&config.GlobalConfig.WebSocket)
    
    // WebSocket连接
    r.GET("/ws", wsManager.HandleWebSocket())
    
    // 广播消息
    r.POST("/broadcast", func(c *gin.Context) {
        message := c.PostForm("message")
        wsManager.Broadcast([]byte(message))
        c.JSON(200, gin.H{"message": "broadcast sent"})
    })
    
    r.Run(":8080")
}

4.2 前端示例

// 连接WebSocket
const ws = new WebSocket('ws://localhost:8080/ws')

// 连接成功
ws.onopen = () => {
    console.log('连接成功')
}

// 接收消息
ws.onmessage = (event) => {
    console.log('收到消息:', event.data)
}

// 发送消息
ws.send('Hello, World!')

4.3 Vue组件示例

<template>
  <div>
    <div>连接状态: {{ isConnected ? '已连接' : '未连接' }}</div>
    <input v-model="message" @keyup.enter="sendMessage">
    <button @click="sendMessage">发送</button>
  </div>
</template>

<script setup>
import { ref, onMounted, onUnmounted } from 'vue'

const ws = ref(null)
const isConnected = ref(false)
const message = ref('')

const connect = () => {
    ws.value = new WebSocket('ws://localhost:8080/ws')
    ws.value.onopen = () => isConnected.value = true
    ws.value.onclose = () => isConnected.value = false
    ws.value.onmessage = (event) => {
        console.log('收到消息:', event.data)
    }
}

const sendMessage = () => {
    if (isConnected.value && message.value) {
        ws.value.send(message.value)
        message.value = ''
    }
}

onMounted(() => connect())
onUnmounted(() => ws.value?.close())
</script>

4.4 消息收发示例

4.4.1 消息结构
// Message 定义消息结构
type Message struct {
    Type     string      `json:"type"`    // 消息类型
    Data     interface{} `json:"data"`    // 消息数据
    ClientID string      `json:"-"`       // 发送者的连接ID
}
4.4.2 发送消息
// 1. 广播消息给所有客户端
err := wsManager.SendMessage("chat", map[string]interface{}{
    "user": "系统",
    "content": "欢迎新用户加入",
})

// 2. 发送消息给指定客户端
err := wsManager.SendMessageToClient(clientAddr, "private", map[string]interface{}{
    "content": "这是一条私信",
    "time": time.Now(),
})

// 3. 发送游戏动作
err := wsManager.SendMessage("game_action", map[string]interface{}{
    "action": "move",
    "position": map[string]int{
        "x": 100,
        "y": 200,
    },
})
4.4.3 接收和处理消息
// 启动消息处理协程
go func() {
    // 获取消息通道
    msgChan := wsManager.GetMessages()
    
    // 循环处理消息
    for msg := range msgChan {
        // 根据消息类型处理
        switch msg.Type {
        case "chat":
            handleChatMessage(msg)
        case "game_action":
            handleGameAction(msg)
        case "private":
            handlePrivateMessage(msg)
        default:
            log.Printf("未知消息类型: %s", msg.Type)
        }
    }
}()

// 处理聊天消息
func handleChatMessage(msg websocket.Message) {
    data, ok := msg.Data.(map[string]interface{})
    if !ok {
        return
    }
    log.Printf("来自 %s 的聊天消息: %v", msg.ClientID, data["content"])
}

// 处理游戏动作
func handleGameAction(msg websocket.Message) {
    data, ok := msg.Data.(map[string]interface{})
    if !ok {
        return
    }
    log.Printf("来自 %s 的游戏动作: %v", msg.ClientID, data["action"])
}
4.4.4 前端发送消息示例
// 1. 发送聊天消息
ws.send(JSON.stringify({
    type: 'chat',
    data: {
        content: '大家好!'
    }
}))

// 2. 发送游戏动作
ws.send(JSON.stringify({
    type: 'game_action',
    data: {
        action: 'move',
        position: {
            x: 100,
            y: 200
        }
    }
}))
4.4.5 前端接收消息示例
ws.onmessage = (event) => {
    try {
        const message = JSON.parse(event.data)
        
        // 根据消息类型处理
        switch (message.type) {
            case 'chat':
                handleChat(message.data)
                break
            case 'game_action':
                handleGameAction(message.data)
                break
            case 'private':
                handlePrivateMessage(message.data)
                break
            default:
                console.log('未知消息类型:', message.type)
        }
    } catch (error) {
        console.error('解析消息失败:', error)
    }
}

// 处理聊天消息
function handleChat(data) {
    console.log(`${data.user}: ${data.content}`)
}

// 处理游戏动作
function handleGameAction(data) {
    console.log('游戏动作:', data.action)
    updateGameState(data.position)
}

// 处理私信
function handlePrivateMessage(data) {
    console.log('收到私信:', data.content)
}

4.5 错误处理示例

// 发送消息时的错误处理
if err := wsManager.SendMessage("chat", data); err != nil {
    log.Printf("发送消息失败: %v", err)
}

// 发送私信时的错误处理
if err := wsManager.SendMessageToClient(clientAddr, "private", data); err != nil {
    if strings.Contains(err.Error(), "client not found") {
        log.Printf("用户已离线: %s", clientAddr)
    } else {
        log.Printf("发送私信失败: %v", err)
    }
}

4.6 消息类型建议

const (
    // 系统消息
    MessageTypeSystem     = "system"      // 系统通知
    MessageTypeError      = "error"       // 错误消息
    
    // 聊天消息
    MessageTypeChat       = "chat"        // 公共聊天
    MessageTypePrivate    = "private"     // 私聊消息
    
    // 游戏消息
    MessageTypeGameAction = "game_action" // 游戏动作
    MessageTypeGameState  = "game_state"  // 游戏状态
    MessageTypeMatch      = "match"       // 匹配相关
)

5. 框架整合

5.1 与Gin框架整合

// 中间件:验证WebSocket连接
func AuthWebSocket() gin.HandlerFunc {
    return func(c *gin.Context) {
        // 验证token
        token := c.Query("token")
        if !validateToken(token) {
            c.AbortWithStatus(http.StatusUnauthorized)
            return
        }
        c.Next()
    }
}

// 使用中间件
r.GET("/ws", AuthWebSocket(), wsManager.HandleWebSocket())

5.2 与Nginx整合

# nginx.conf
http {
    map $http_upgrade $connection_upgrade {
        default upgrade;
        '' close;
    }
    
    upstream websocket {
        server 127.0.0.1:8080;
    }
    
    server {
        listen 80;
        server_name example.com;
        
        # WebSocket代理
        location /ws {
            proxy_pass http://websocket;
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection $connection_upgrade;
            proxy_set_header Host $host;
        }
        
        # 其他HTTP请求
        location / {
            proxy_pass http://127.0.0.1:8080;
            proxy_set_header Host $host;
            proxy_set_header X-Real-IP $remote_addr;
        }
    }
}

6. 部署说明

6.1 服务器要求

  • 支持WebSocket的现代浏览器
  • Go 1.16+
  • Nginx 1.16+(如果使用)

6.2 部署步骤

  1. 编译Go程序
go build -o server cmd/main.go
  1. 配置Nginx(如果使用)
# 复制nginx配置
sudo cp nginx.conf /etc/nginx/conf.d/websocket.conf

# 重启Nginx
sudo systemctl restart nginx
  1. 运行服务
./server

6.3 注意事项

  1. 连接管理

    • 定期清理断开的连接
    • 实现重连机制
    • 控制连接数量
  2. 安全性

    • 添加连接认证
    • 限制消息大小
    • 添加消息验证
  3. 性能优化

    • 使用消息队列
    • 控制广播频率
    • 添加负载均衡
  4. 监控

    • 记录连接数
    • 监控消息流量
    • 错误日志记录

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2273378.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

LInux单机安装Redis

1. 安装gee工具包 由于Redis是基于c语言编写的所以安装的时候需要先安装gee以及gcc的依赖,yum云用不了可以看一下这个 linux 替换yum源镜像_更换yum镜像源-CSDN博客 yum install -y gcc tcl 2. 添加redis的压缩包 3. 上传到Linux 上传到 /usr/local/src 目录、这个目录一般用于…

VSCode 使用鼠标滚轮控制字体

一、 文件 | 首选项 | 设置 二、单击在 settings.json中编辑 "editor.mouseWheelZoom": true 注注注意&#xff1a;保存哦&#xff01;ctrlS 三、测试 按住ctrl鼠标滚轮&#xff0c;控制字体大小

enzymejest TDD与BDD开发实战

一、前端自动化测试需要测什么 1. 函数的执行逻辑&#xff0c;对于给定的输入&#xff0c;输出是否符合预期。 2. 用户行为的响应逻辑。 - 对于单元测试而言&#xff0c;测试粒度较细&#xff0c;需要测试内部状态的变更与相应函数是否成功被调用。 - 对于集成测试而言&a…

TCP通信原理学习

TCP三次握手和四次挥手以及为什么_哔哩哔哩_bilibili

空间不足导致Oracle集群内存使用率暴增

一、现象 操作系统内存使用率告警&#xff0c;已达到98%,&#xff0c;告警内容如下&#xff1a; 【全景监控&#xff1a;Oracle主机内存使用监控】 【主机名】&#xff1a;XXXXX11 【主机IP】主机IP&#xff1a;*.126.15 【告警内容】当前内存使用率为98.9%&#xff0c;超警…

嵌入式入门Day38

C Day1 第一个C程序C中的输入输出输出操作coutcin练习 命名空间使用方法自定义命名空间冲突问题 C对字符串的扩充C风格字符串的使用定义以及初始化C风格字符串与C风格字符串的转换C风格的字符串的关系运算常用的成员变量输入方法 布尔类型C对堆区空间使用的扩充作业 第一个C程序…

【JMM】Java 内存模型

&#x1f970;&#x1f970;&#x1f970;来都来了&#xff0c;不妨点个关注叭&#xff01; &#x1f449;博客主页&#xff1a;欢迎各位大佬!&#x1f448; 文章目录 1. 前言2. JMM 内存模型内容3. JMM 内存模型简单执行示意图 ⚠️ 不要与 JVM 内存分布混为一谈论&#xff0c…

SEO新革命:如何通过Search Everywhere优化全面打破搜索壁垒

谷歌不再总是人们寻求答案的首选之地。他们越来越多地转向社交媒体、YouTube、亚马逊和 ChatGPT。这些平台本身已经成为搜索引擎。 因此&#xff0c;SEO 需要发展。仅靠搜索 “引擎” 优化已经不够了。品牌需要优化其在每个平台上的自然存在。 您需要一种新型的 SEO&#xff…

Spring 设计模式:经典设计模式

Spring 设计模式&#xff1a;经典设计模式 引言 Spring 框架广泛使用了经典设计模式。 这些模式在 Spring 内部发挥着重要作用。 通过理解这些设计模式在 Spring 中的应用&#xff0c;开发者可以更深入地掌握 Spring 框架的设计哲学和实现细节。 经典设计模式 控制反转&am…

“AI 视频图像识别系统,开启智能新视界

咱老百姓现在的生活啊&#xff0c;那是越来越离不开高科技了&#xff0c;就说这 AI 视频图像识别系统&#xff0c;听起来挺高大上&#xff0c;实际上已经悄无声息地融入到咱们日常的方方面面&#xff0c;给咱带来了超多便利。 先讲讲安防领域吧&#xff0c;这可是 AI 图像识别的…

开源AI智能名片2+1链动模式S2B2C商城小程序在商业流量获取中的应用研究

摘要&#xff1a; 随着互联网技术的迅猛发展&#xff0c;商业流量的获取已成为企业市场竞争中的关键环节。传统意义上的“客流量”在互联网语境下被赋予了新的内涵&#xff0c;即“商业流量”&#xff0c;其本质依然指向用户。在当前线上线下融合的商业环境中&#xff0c;流量…

【蓝桥杯研究生组】第14届Java试题答案整理

试题链接&#xff1a;链接 A题 满足条件的答案有&#xff1a;35813116 public class TianShu {public static void main(String[] args) {int ans 0;// 2000.1.1 - 2000000.1.1// 年份是月份的倍数&#xff0c;也是日的倍数for (int year2000; year<2000000; year) {for …

c/c++ 里的进程间通信 , 管道 pipe 编程举例

&#xff08;1&#xff09;以下是一个网上的使用 pipe 编程的范例&#xff1a; #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <sys/types.h> #include <sys/wait.h>int main() {int pipefd…

RK3588+FPGA全国产异步LED显示屏控制卡/屏幕拼接解决方案

RK3588FPGA核心板采用Rockchip RK3588新一代旗舰 级八核64位处理器&#xff0c;支持8K视频编解码&#xff0c;多屏4K输出&#xff0c;可实现12屏联屏拼接、同显、异显&#xff0c;适配多种操作系统&#xff0c;广泛适用于展览展示、广告内容投放、新零售、商超等领域实现各种媒…

uniapp使用chooseLocation安卓篇

本文章全部以高德地图为例 代码 <view class"bottom"><button click"choose">定位</button> </view> choose() {uni.chooseLocation({success: function(res) {console.log(位置名称&#xff1a; res.name);console.log(详细地…

flutter 专题三十三 Flutter 重构去哪儿QTalk

QTalk 是去哪儿网内部的一个 IM 沟通工具&#xff0c;同时集成了很多内部的系统&#xff0c;比如 OA 审批&#xff0c;门禁打卡&#xff0c;请假审批&#xff0c;预定会议室&#xff0c;驼圈&#xff08;驼厂朋友圈&#xff09;等功能&#xff1b;方便内部办公沟通、交流的同时…

任务调度之Quartz(二):Quartz体系结构

1、Quartz 体系结构 由上一篇的Quartz基本使用可以发现&#xff0c;Quartz 主要包含一下几种角色&#xff1a; 1&#xff09;Job&#xff1a;也可以认为是JobDtetail&#xff0c;表示具体的调度任务 2&#xff09;Trigger&#xff1a;触发器&#xff0c;用于定义任务Job出发执行…

141.环形链表 142.环形链表II

141.环形链表 & 142.环形链表II 141.环形链表 思路&#xff1a;快慢指针 or 哈希表 快慢指针代码&#xff1a; class Solution { public:bool hasCycle(ListNode *head) {if(headnullptr||head->nextnullptr)return false;ListNode *fasthead->next; //不能设置成…

软件项目体系建设文档,项目开发实施运维,审计,安全体系建设,验收交付,售前资料(word原件)

软件系统实施标准化流程设计至关重要&#xff0c;因为它能确保开发、测试、部署及维护等各阶段高效有序进行。标准化流程能减少人为错误&#xff0c;提升代码质量和系统稳定性。同时&#xff0c;它促进了团队成员间的沟通与协作&#xff0c;确保项目按时交付。此外&#xff0c;…

uniapp-vue3 实现, 一款带有丝滑动画效果的单选框组件,支持微信小程序、H5等多端

采用 uniapp-vue3 实现, 是一款带有丝滑动画效果的单选框组件&#xff0c;提供点状、条状的动画过渡效果&#xff0c;支持多项自定义配置&#xff0c;适配 web、H5、微信小程序&#xff08;其他平台小程序未测试过&#xff0c;可自行尝试&#xff09; 可到插件市场下载尝试&…