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

【go从零单排】Stateful Goroutines(有状态的 goroutines)

挪威特罗姆瑟夜景

🌈Don’t worry , just coding!
内耗与overthinking只会削弱你的精力,虚度你的光阴,每天迈出一小步,回头时发现已经走了很远。

📗概念

在 Go 中,有状态的 goroutines(Stateful Goroutines)是指那些能够保留状态信息的 goroutines。这种设计模式通常用于需要在多个调用之间维护状态的场景,例如处理请求、管理连接、维护计数器等。

💻代码

package main

//导入math/rand 用于生成随机数。
//sync/atomic 提供原子操作的支持。
import (
	"fmt"
	"math/rand"
	"sync/atomic"
	"time"
)

// readOp 结构体用于表示读操作,包含一个键 key 和一个响应通道 resp。
type readOp struct {
	key  int
	resp chan int
}

// readOp 结构体用于表示读操作,包含一个键 key 和一个响应通道 resp。
type writeOp struct {
	key  int
	val  int
	resp chan bool
}

func main() {
	//声明两个无符号整型变量 readOps 和 writeOps,用于计数读写操作。
	var readOps uint64
	var writeOps uint64
	//创建两个通道 reads 和 writes,分别用于接收读操作和写操作。
	reads := make(chan readOp)
	writes := make(chan writeOp)

	go func() {
		//启动一个 goroutine,维护一个状态 state,这是一个映射,存储键值对。
		var state = make(map[int]int)
		for {
			//使用 select 语句监听 reads 和 writes 通道:
			select {
			//如果接收到读操作,从 state 中读取对应的值并通过 resp 通道返回。
			case read := <-reads:
				read.resp <- state[read.key]
			//如果接收到写操作,将值写入 state 中,并通过 resp 通道返回操作成功的信号。
			case write := <-writes:
				state[write.key] = write.val
				write.resp <- true
			}
		}
	}()
	//启动 100 个 goroutine
	for r := 0; r < 100; r++ {
		go func() {
			for {
				//创建一个随机键的 readOp 实例并发送到 reads 通道。
				read := readOp{
					key:  rand.Intn(5),
					resp: make(chan int)}
				reads <- read
				<-read.resp
				//创建一个随机键的 readOp 实例并发送到 reads 通道。
				atomic.AddUint64(&readOps, 1)
				//每次循环后休眠 1 毫秒。
				time.Sleep(time.Millisecond)
			}
		}()
	}
	//启动 10 个 goroutine
	for w := 0; w < 10; w++ {
		go func() {
			for {
				//创建一个随机键和随机值的 writeOp 实例并发送到 writes 通道。
				write := writeOp{
					key:  rand.Intn(5),
					val:  rand.Intn(100),
					resp: make(chan bool)}
				writes <- write
				<-write.resp
				//等待响应并通过原子操作增加 writeOps 的计数。
				atomic.AddUint64(&writeOps, 1)
				//每次循环后休眠 1 毫秒。
				time.Sleep(time.Millisecond)
			}
		}()
	}

	time.Sleep(time.Second)
	//使用 atomic.LoadUint64 获取最终的读写操作计数并打印。
	readOpsFinal := atomic.LoadUint64(&readOps)
	fmt.Println("readOps:", readOpsFinal)
	writeOpsFinal := atomic.LoadUint64(&writeOps)
	fmt.Println("writeOps:", writeOpsFinal)
}
//输出
//readOps: 81299
//writeOps: 8163

有状态的 Goroutines 的特点

  • 状态管理:有状态的 goroutines 可以在其生命周期内维护状态信息,这样它们可以根据之前的操作或输入做出决策。
  • 并发安全:在并发环境中,状态的管理需要确保安全性,通常使用通道、互斥锁或其他同步机制来保护共享状态。
  • 封装性:将状态和行为封装在 goroutine 内部,使得外部无法直接访问状态,从而增强了模块化和封装性。

🔍理解

  • 并发读写:代码实现了一个简单的并发读写操作,使用通道来协调读写请求。
  • 原子操作:使用 sync/atomic 包中的原子操作来安全地更新读写计数,避免数据竞争。
  • 随机访问:读和写操作都是基于随机生成的键值,使得操作更加多样化。
  • 无锁设计:通过通道和 goroutine 的组合,避免了传统的锁机制,简化了并发控制。

💪无人扶我青云志,我自踏雪至山巅。
在这里插入图片描述


http://www.kler.cn/a/392352.html

相关文章:

  • java项目之校园管理系统的设计与实现(源码+文档)
  • 在Mac电脑上搭建Gradle
  • SQL—Group_Concat函数用法详解
  • 深入理解 Java 接口的回调机制
  • 【Unity3D】UGUI Canvas画布渲染流程
  • 若依中Feign调用的具体使用(若依微服务版自身已集成openfeign依赖,并在此基础上定义了自己的注解)
  • 2024 年 Postman 网页版使用图文教程
  • Tomcat(6) 什么是Servlet容器?
  • 单例模式和适配器模式的简单介绍
  • [ACTF2020 新生赛]Upload 1--详细解析
  • JVM(一、基础知识)
  • 7. 基于 Redis 实现分布式锁
  • 基于Java Web的传智播客crm企业管理系统的设计与实现
  • 公开仓库改私有再配置公钥后Git拉取仍需要输入用户名的问题
  • 普通用户切换到 root 用户不需要输入密码配置(Ubuntu20)
  • vxe-table 3.10+ 进阶高级用法(一),根据业务需求自定义实现筛选功能
  • 【软考】系统架构设计师-计算机系统基础(2):操作系统
  • 【Linux】Linux 命令awk和sed的简单介绍
  • Vestar:AI造神邸,颠覆Meme叙事的新范式
  • 45.第二阶段x86游戏实战2-hook监控实时抓取游戏lua
  • 【python GUI编码入门-24】使用Tkinter构建一个简单的音乐播放器
  • 【Linux:IO多路复用(select函数)
  • 新的服务器Centos7.6 安装基础的环境配置(新服务器可直接粘贴使用配置)
  • 数据集标注txt文件读取小工具
  • # 如何查看 Ubuntu 版本?
  • Java | Leetcode Java题解之第559题N叉树的最大深度