【Go】三、Go并发编程

news2025/1/18 7:37:01

并发编程

我们主流的并发编程思路一般有:多进程、多线程

但这两种方式都需要操作系统介入,进入内核态,是十分大的时间开销

由此而来,一个解决该需求的技术出现了:用户级线程,也叫做 绿程、轻量级线程、协程

python - asyncio、java - netty22111111111111115

由于 go 语言是 web2.0 时代发展起来的语言,go语言没有多线程和多进程的写法,其只有协程的写法 golang - goroutine

func Print() {
	fmt.Println("打印印")
}

func main() {
	go Print()
}

我们可以使用这种方式来进行并发编程,但这个程序里要注意,我们主程序在确定完异步之后结束,会立即让程序退出,这就导致我们并发的子线程没来得及执行就退出了。

我们可以增加一个Sleep来让主线程让出资源,等待子线程执行完毕再进行操作

func Print() {
	for {
		time.Sleep(time.Second)
		fmt.Println("打印印")
	}

}

func main() {
	go Print()
	for {
		time.Sleep(time.Second)
		fmt.Println("主线程")
	}
}

另外的,Go 语言协程的一个巨大优势是 可以打开成百上千个协程,协助程序效率的提升

要注意一个问题:

多进程的切换十分浪费时间,且及其浪费系统资源

多线程的切换也很浪费时间,但其解决了浪费系统资源的问题

协程既解决了切换浪费时间的问题,也解决了浪费系统资源的问题

Go语言仅支持协程

Go语言中,协程的调度(gmp机制):

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

用户新建的协程(Goroutine)会被加入到同一调度器中等待运行,若调度器满,则会将调度器中一半的 G 加入到全局队列中,其他 P 若没有 G 则会从其他 P 中偷取一半的 G ,若所有的都满,则会新建 M 进行处理

P 的数量是固定的

注意 M 和 P 不是永远绑定的,当一个 P 现在绑定的 M 进入了阻塞等情况,P 会自动去寻找空闲的 M 或创建新的 M 来绑定

子 goroutine 如何通知到主 goroutine 其运行状态?也就是我们主协程要知道子协程运行完毕之后再进行进一步操作,也就是 (wait)

func main() {

	// 定义 sync.Group 类型的变量用于控制goroutine的状态
	var wg sync.WaitGroup

	for i := 0; i < 100; i++ {
		go func(i int) {
			wg.Add(1)       // 每次启动一个协程都要开启一个计数器
			defer wg.Done() // 每次结束之前都要让计数器 -1
			fmt.Println("这次打印了:" + strconv.Itoa(i))
		}(i)
	}
	wg.Wait()
	fmt.Println("结束..................................")
}

Go语言中的锁

互斥锁

我们看下面这个程序:

var total int
var wg sync.WaitGroup

func add() {
	defer wg.Done()
	for i := 0; i <= 100000; i++ {
		total += 1
	}
}

func sub() {
	defer wg.Done()
	for i := 0; i <= 100000; i++ {
		total -= 1
	}
}

func main() {
	wg.Add(2)
	go add()
	go sub()
	wg.Wait()
	fmt.Println(total)
	/*
		-100001
		100001
		68595
		...
	*/
}

我们发现这个程序的结果不可预知,这是因为 a 的操作分为三步:

取得 a 的值、执行 a 的计算操作、写入 a 的值,这三步不是原子性的,如果发生了交叉,则一个数准备写入时发生协程切换,这时后面再做再多的操作,也会被这次切换屏蔽掉,最终写入这一个数的结果,由此可知,这种非原子性操作共享数据的模式是不可预知结果的。

那么,我们就需要加锁,像下面这样

var total int
var wg sync.WaitGroup
var lock sync.Mutex
var lockc = &lock

func add() {
	defer wg.Done()
	for i := 0; i <= 100000; i++ {
		lock.Lock() // 加锁,直至见到自己这把锁的 Unlock() 方法之前,令这中间的方法都为原子性的
		total += 1
		lock.Unlock() // 解锁,配合 Lock() 方法使用
	}
}

func sub() {
	defer wg.Done()
	for i := 0; i <= 100000; i++ {
		lockc.Lock()
		total -= 1
		lockc.Unlock()
	}
}

func main() {
	wg.Add(2)
	go add()
	go sub()
	wg.Wait()

	fmt.Println(total)
	fmt.Printf("%p\n", &lock)
	fmt.Printf("%p\n", &(*lockc))
}

注意,上面这个程序不仅演示了加锁,还演示了,浅拷贝不影响加锁的情况

另外,我们也可以使用 automic 对简单的数值计算进行加锁

var total int32
var wg sync.WaitGroup
var lock sync.Mutex
var lockc = &lock

func add() {
	defer wg.Done()
	for i := 0; i <= 100000; i++ {
		atomic.AddInt32(&total, 1)
	}
}

func sub() {
	defer wg.Done()
	for i := 0; i <= 100000; i++ {
		atomic.AddInt32(&total, -1)
	}
}

func main() {
	wg.Add(2)
	go add()
	go sub()
	wg.Wait()
	fmt.Println(total)
}

读写锁

读写锁就是:允许同时读,不允许同时写,不允许同时读写

func main() {
	var num int
	var rwlock sync.RWMutex // 定义一个读写锁
	var wg sync.WaitGroup   // 定义等待处理器

	wg.Add(2)

	go func() {
		defer wg.Done()
		rwlock.Lock() // 写锁
		defer rwlock.Unlock()
		num = 12
	}()

	// 同步处理器,这里是简便处理
	time.Sleep(1)

	go func() {
		defer wg.Done()
		rwlock.RLock()
		defer rwlock.RUnlock()
		fmt.Println(num)
	}()

	wg.Wait()

}

一个简单的测试

func main() {
	var rwlock sync.RWMutex // 定义一个读写锁
	var wg sync.WaitGroup   // 定义等待处理器

	wg.Add(6)

	go func() {
		time.Sleep(time.Second)
		defer fmt.Println("释放写锁,可以进行读操作")
		defer wg.Done()
		rwlock.Lock() // 写锁
		defer rwlock.Unlock()
		fmt.Println("得到写锁,停止读操作")
		time.Sleep(time.Second * 5)
	}()

	for i := 0; i < 5; i++ {
		go func() {
			defer wg.Done()
			for {
				rwlock.RLock()
				fmt.Println("得到读锁,进行读操作")
				time.Sleep(time.Millisecond * 500)
				rwlock.RUnlock()
			}
		}()
	}

	wg.Wait()
	/**
	得到读锁,进行读操作

	得到写锁,停止读操作
	释放写锁,可以进行读操作
	
	得到读锁,进行读操作
	得到读锁,进行读操作
	得到读锁,进行读操作
	*/

}

通信

Go 语言中对于并发场景下的通信,秉持以下理念:

不要通过共享内存来通信,要通过通信实现共享内存

其他语言都是用一个共享的变量来实现通信,或者消息队列,Go语言就希望实现队列的机制

	var msg chan string //定义一个用于传递 string 的 channel

	// 创建一个缓冲区大小为 1 的 channel
	// 只有 有缓冲区的 channel 才可以暂存数据
	msg = make(chan string, 1)

	msg <- "data"
	data := <-msg
	fmt.Println(data)

只有 goroutine 中才可以使用缓冲区大小为 0 的channel

func main() {
	var msg chan string //定义一个用于传递 string 的 channel

	// 创建一个缓冲区大小为 1 的 channel
	// 只有 有缓冲区的 channel 才可以暂存数据
	msg = make(chan string, 0)

	go func(msg chan string) {
		data := <-msg
		fmt.Println(data)
	}(msg)

	msg <- "data"
	time.Sleep(time.Second * 3)
}

这时由于 go 语言 channel 中的 happen-before 机制,该机制保证了 就算先 receiver 也会被 goroutine 挂起,等待 sender 完成之后再进行 receiver 的具体执行

go 语言中,channel 的应用场景十分广泛,包括:

  • 信息传递、消息过滤
  • 信号广播
  • 事件订阅与广播
  • 任务分发
  • 结果汇总
  • 并发控制
  • 同步异步

Go 语言的消息接收问题

func main() {
	var msg chan string //定义一个用于传递 string 的 channel

	// 创建一个缓冲区大小为 1 的 channel
	// 只有 有缓冲区的 channel 才可以暂存数据
	msg = make(chan string, 0)

	go func(msg chan string) {
		// 注意这里,每一个接收消息的变量只能接收到一个消息,若有多条消息同时发送,则无法接收
		data := <-msg
		fmt.Println(data)

		math := <-msg
		fmt.Println(math)
	}(msg)

	msg <- "data"
	msg <- "math"
	time.Sleep(time.Second * 3)

}

如果我们不知道消息会发送来多少,可以使用 for-range 进行监听:

func main() {
	var msg chan string //定义一个用于传递 string 的 channel

	// 创建一个缓冲区大小为 1 的 channel
	// 只有 有缓冲区的 channel 才可以暂存数据
	msg = make(chan string, 2)

	go func(msg chan string) {
		// 若我们不确定有多少消息会过来,我们可以使用 for-range 进行循环验证
		for data := range msg {
			fmt.Println(data)
		}
	}(msg)

	msg <- "data"
	msg <- "math"
	time.Sleep(time.Second * 3)

}
close(msg) // 关闭队列,监听队列的 goroutine 会立刻退出

关闭了的 channel 不能再存储数据,但可以进行数据的取出操作

上面我们所接触的 channel 都是双向的 channel 即这个channel 对应的goroutine 既可以从里面读数据,也可以向里面写数据,这种不符合我们程序,一个程序只做它对应的一个功能,这一程序设计思路

创建单向 channel:

	var ch1 chan int       // 这是一个双向 channel
	var ch2 chan<- float64 // 这是一个只能写入 float64 类型数据的单向 channel
	var ch3 <-chan int     // 这是一个只能从 存储int型 channel中读取数据的单向channel
	c := make(chan int, 3)     // 创建一个双向 channel
	var send chan<- int = c    // 将 c channel 的写入能力赋予给 send,使其成为一个单向发送的 channel (生产者)
	var receive <-chan int = c // 将c channel 的读取能力赋予给receive,使其成为一个单向接收的 channel (消费者)

经典例子:

/**
经典:
使用两个 goroutine 交替打印:12AB34CD56EF78GH910IJ1112.....YZ2728
*/

var number, letter = make(chan bool), make(chan bool)

func printNum() {
	// 这里是等待接收消息,若消息接收不到,则该协程会始终阻塞在这个位置
	i := 1
	for {
		<-number
		fmt.Printf("%d%d", i, i+1)
		i += 2
		letter <- true
	}
}

func printLetter() {
	i := 0
	str := "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
	for {
		<-letter
		fmt.Print(str[i : i+2])
		number <- true
		if i <= 23 {
			i += 2
		} else {
			return
		}

	}

}

func main() {
	go printNum()
	go printLetter()
	number <- true
	time.Sleep(time.Second * 20)
}

使用 select 对 goroutine 进行监控

// 使用 struct{} 作为传入的信息,由于这个 struct{} 占用内存空间较小,这是一种常见的传递方式

func g1(ch chan struct{}) {
	time.Sleep(time.Second * 3)
	ch <- struct{}{}
}

func g2(ch chan struct{}) {
	time.Sleep(time.Second * 3)
	ch <- struct{}{}
}

func main() {
	g1Channel := make(chan struct{})
	g2Channel := make(chan struct{})

	go g1(g1Channel)
	go g2(g2Channel)

	// 注意这里只要有一个能取到值则 select 结果则结束
	select {
	// 若 g1Channel 中能取到值
	case <-g1Channel:
		fmt.Println("g1 done")
	// 若 g1Channel 中能取到值
	case <-g2Channel:
		fmt.Println("g2 done")
	}
}

这里若所有的 goroutine 都就绪了,则 select 执行哪个是随机的,为的是防止某个 goroutine 一直被优先执行导致的另一个 goroutine 饥饿

超时机制:

// 使用 struct{} 作为传入的信息,由于这个 struct{} 占用内存空间较小,这是一种常见的传递方式

func g1(ch chan struct{}) {
	time.Sleep(time.Second * 3)
	ch <- struct{}{}
}

func g2(ch chan struct{}) {
	time.Sleep(time.Second * 3)
	ch <- struct{}{}
}

func main() {
	g1Channel := make(chan struct{})
	g2Channel := make(chan struct{})

	go g1(g1Channel)
	go g2(g2Channel)

	timeChannel := time.NewTimer(5 * time.Second)

	for {
		// 注意这里只要有一个能取到值则 select 结果则结束
		select {
		// 若 g1Channel 中能取到值
		case <-g1Channel:
			fmt.Println("g1 done")
		// 若 g1Channel 中能取到值
		case <-g2Channel:
			fmt.Println("g2 done")
		case <-timeChannel.C: // timeChannel.C 是获取我们创建的 channel 的方法
			fmt.Println("time out")
			return
		}
	}
}

context

使用 WithCancel() 引入手动终止进程的功能

func cpuIInfo(ctx context.Context) {
	defer wg.Done()
	for {
		select {
		case <-ctx.Done(): // 本质还是一个 channel
			fmt.Println("程序退出执行...........")
			return
		default:
			time.Sleep(1 * time.Second)
			fmt.Println("CPUUUUUUUUUUUUUU")
		}
	}
}

func main() {
	/**
	这里有一个问题,我们可以将以 context.Background() 为参数的 context 是作为最上层的父 context
	所有以其他 context 为参数的 context 都是他的子 context
	只要父 context 调用了 cancel() 则其所有的子 context 都会停止
	*/
	ctxParent, cancel := context.WithCancel(context.Background())
	ctxChild, _ := context.WithCancel(ctxParent)

	wg.Add(1)
	go cpuIInfo(ctxChild)
	time.Sleep(5 * time.Second)
	cancel() // 这个方法会直接向context channel 中传入一个对象,令channel停止
	wg.Wait()
}

使用 WthTimeout() 来自动引入超时退出机制

var wg sync.WaitGroup

func cpuIInfo(ctx context.Context) {
	defer wg.Done()
	for {
		select {
		case <-ctx.Done(): // 本质还是一个 channel
			fmt.Println("程序退出执行...........")
			return
		default:
			time.Sleep(1 * time.Second)
			fmt.Println("CPUUUUUUUUUUUUUU")
		}
	}
}

func main() {
	/**
	这里有一个问题,我们可以将以 context.Background() 为参数的 context 是作为最上层的父 context
	所有以其他 context 为参数的 context 都是他的子 context
	只要父 context 调用了 cancel() 则其所有的子 context 都会停止
	*/
	ctxParent, _ := context.WithTimeout(context.Background(), 6*time.Second)
	ctxChild, _ := context.WithCancel(ctxParent)

	wg.Add(1)
	go cpuIInfo(ctxChild)
	wg.Wait()
}

WithDeadline() 是指定某个时间点,在某个时间点的时候进行执行

WithValue() 则会向 context 中传递一个数据,我们可以在子 goroutine 中调用这个数据

var wg sync.WaitGroup

func cpuIInfo(ctx context.Context) {
	defer wg.Done()
	for {
		select {
		case <-ctx.Done(): // 本质还是一个 channel
			fmt.Println("程序退出执行...........")
			return
		default:
			time.Sleep(1 * time.Second)
			fmt.Printf("%s", ctx.Value("LeaderID"))
			fmt.Println("CPUUUUUUUUUUUUUU")
		}
	}
}

func main() {
	/**
	这里有一个问题,我们可以将以 context.Background() 为参数的 context 是作为最上层的父 context
	所有以其他 context 为参数的 context 都是他的子 context
	只要父 context 调用了 cancel() 则其所有的子 context 都会停止
	*/
	ctxParent, _ := context.WithTimeout(context.Background(), 6*time.Second)
	ctxChild := context.WithValue(ctxParent, "LeaderID", "00001") // 注意 WithValue 方法只有一个返回值

	wg.Add(1)
	go cpuIInfo(ctxChild)
	wg.Wait()
}

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

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

相关文章

Leetcode—42. 接雨水【困难】

2024每日刷题&#xff08;112&#xff09; Leetcode—42. 接雨水 空间复杂度为O(n)的算法思想 实现代码 class Solution { public:int trap(vector<int>& height) {int ans 0;int n height.size();vector<int> l(n);vector<int> r(n);for(int i 0; …

《学成在线》微服务实战项目实操笔记系列(P1~P62)【上】

《学成在线》项目实操笔记系列【上】&#xff0c;跟视频的每一P对应&#xff0c;全系列12万字&#xff0c;涵盖详细步骤与问题的解决方案。如果你操作到某一步卡壳&#xff0c;参考这篇&#xff0c;相信会带给你极大启发。同时也欢迎大家提问与讨论&#xff0c;我会尽力帮大家解…

RK3568笔记十二:Zlmedia拉流显示测试

若该文为原创文章&#xff0c;转载请注明原文出处。 Zlmediakit功能很强大&#xff0c;测试一下拉流&#xff0c;在通过解码显示。 一、环境 1、平台&#xff1a;rk3568 2、开发板:ATK-RK3568正点原子板子 3、环境&#xff1a;buildroot 测试的代码在GitHub - airockchip/…

ubuntu20安装mongodb

方法一&#xff1a;直接安装(命令是直接从mongo官网Install MongoDB Community Edition on Ubuntu — MongoDB Manual复制的&#xff09; cat /etc/lsb-release sudo apt-get install -y gnupg curl curl -fsSL https://www.mongodb.org/static/pgp/server-7.0.asc | \sudo gp…

VS Code中主程序C文件引用了另一个.h头文件,编译时报错找不到函数

目录 一、问题描述二、问题原因三、解决方法四、扩展五、通过CMake进行配置 一、问题描述 VS Code中主程序C文件引用了另一个.h头文件&#xff0c;编译时报错找不到函数 主程序 main.c #include <stdio.h> #include "sumaa.h"int main(int, char**){printf(&q…

秘塔科技推出AI搜索产品「秘塔AI搜索」

近日&#xff0c;国内一家人工智能科技公司&#xff08;秘塔科技&#xff09;推出了一款AI搜索产品——秘塔AI搜索&#xff0c;能够大幅提升搜索效率&#xff0c;解决日常生活、工作学习等场景中遇到的各类搜索需求。 秘塔AI搜索官网&#xff1a;https://metaso.cn/ 相较于传统…

Java 学习和实践笔记(2)

今天的学习进度&#xff1a; 注册并下载安装好了Java 8&#xff0c;之后进行以下配置。 1&#xff09;path 是一个常见的环境变量&#xff0c;它告诉系统除了在当前的目标下妹寻找此程序外&#xff0c;还可以到path指定的目录下找。 2&#xff09;Java Home 为以后其他的软…

FastAdmin西陆房产系统(xiluHouse)全开源

应用介绍 一款基于FastAdminThinkPHPUniapp开发的西陆房产管理系统&#xff0c;支持小程序、H5、APP&#xff1b;包含房客、房东(高级授权)、经纪人(高级授权)三种身份。核心功能有&#xff1a;新盘销售、房屋租赁、地图找房、房源代理(高级授权)、在线签约(高级授权)、电子合同…

MATLAB环境下用于提取冲击信号的几种解卷积方法

卷积混合考虑了信号的时延&#xff0c;每一个单独源信号的时延信号都会和传递路径发生一 次线性瞬时混合&#xff1b;解卷积的过程就是找一个合适的滤波器&#xff0c;进行反卷积运算&#xff0c;得到源信号的近似解。 声音不可避免的会发生衍射、反射等现象&#xff0c;所以&…

(注解配置AOP)学习Spring的第十七天

基于注解配置的AOP 来看注解式开发 : 先把目标与通知放到Spring里管理 : Service("userService") public class UserServiceImpl implements UserService {Overridepublic void show1() {System.out.println("show1......");}Overridepublic void show2…

Elasticsearch:使用 LangChain 文档拆分器进行文档分块

使用 Elasticsearch 嵌套密集向量支持 这个交互式笔记本将&#xff1a; 将模型 “sentence-transformers__all-minilm-l6-v2” 从 Hugging Face 加载到 Elasticsearch ML Node 中使用 LangChain 分割器将段落分块成句子&#xff0c;并使用嵌套密集向量将它们索引到 Elasticse…

【RL】Bellman Equation (贝尔曼等式)

Lecture2: Bellman Equation State value 考虑grid-world的单步过程&#xff1a; S t → A t R t 1 , S t 1 S_t \xrightarrow[]{A_t} R_{t 1}, S_{t 1} St​At​ ​Rt1​,St1​ t t t, t 1 t 1 t1&#xff1a;时间戳 S t S_t St​&#xff1a;时间 t t t时所处的sta…

基于蒙特卡洛的电力系统可靠性分析matlab仿真,对比EDNS和LOLP

目录 1.课题概述 2.系统仿真结果 3.核心程序与模型 4.系统原理简介 1.课题概述 电力系统可靠性是指电力系统按可接受的质量标准和所需数量不间断地向电力用户供应电力和电能量的能力的量度&#xff0c;包括充裕度和安全性两个方面。发电系统可靠性是指统一并网的全部发电机…

【RT-DETR有效改进】重参数化模块DiverseBranchBlock助力特征提取(附代码 + 修改教程)

&#x1f451;欢迎大家订阅本专栏&#xff0c;一起学习RT-DETR&#x1f451; 一、本文介绍 本文给大家带来的是改进机制是一种替换多元分支模块&#xff08;Diverse Branch Block&#xff09;&#xff0c;Diverse Branch Block (DBB) 是一种用于增强卷积神经网络性能的结构…

Vue前端框架--Vue工程项目问题总结{脚手架 Vue-cli}

Vue脚手架部署问题总结 我所遇到的一共两大问题 只有先执行npm install之后 才能run serve 否则会报错 vue-cli-serve不是内部或者外部的命令&#xff0c;也不是可运行的程序或者批处理文件的错误 1. 运行npm install会报错 2. 运行npm run serve报错 nodejs官网为 https://no…

算法之双指针系列1

目录 一&#xff1a;双指针的介绍 1&#xff1a;快慢指针 2&#xff1a;对撞指针 二&#xff1a;对撞指针例题讲述 一&#xff1a;双指针的介绍 在做题中常用两种指针&#xff0c;分别为对撞指针与快慢指针。 1&#xff1a;快慢指针 简称为龟兔赛跑算法&#xff0c;它的基…

Unity引擎学习笔记之【动画层操作】

动画层Animation Layer 一、动画器的三个基本状态 1. Any State&#xff08;任意状态&#xff09; “Any State”&#xff08;任意状态&#xff09;&#xff1a;这个状态可以用来连接多个状态机的任意状态转换。在动画控制器中&#xff0c;你可以使用“Any State”作为过渡条…

前端架构: 从vue-cli探究脚手架原理

从使用角度理解什么是脚手架 脚手架本质是一个操作系统的客户端 在终端中去执行一个命令&#xff0c;这个命令本身它就是一个客户端我们其实可以把脚手架理解为操作系统的一个客户端通过命令去执行它的时候&#xff0c;这个命令往往是这样的一个构造&#xff0c;如下 比如&…

CTFshow web(php命令执行 37-40)

?ceval($_GET[shy]);&shypassthru(cat flag.php); #逃逸过滤 ?cinclude%09$_GET[shy]?>&shyphp://filter/readconvert.base64-encode/resourceflag.php #文件包含 ?cinclude%0a$_GET[cmd]?>&cmdphp://filter/readconvert.base64-encode/…

npm 下载报错

报错信息 : 证书过期 (CERT_HAS_EXPIRED) D:\Apps\nodejs-v18.16.1\npx.cmd --yes create-next-app"latest" . --ts npm ERR! code CERT_HAS_EXPIRED npm ERR! errno CERT_HAS_EXPIRED npm ERR! request to https://registry.npm.taobao.org/create-next-app failed…