网站首页 > 技术文章 正文
对并发的处理还有 3 种思路:
- 多核并行化
- 出让时间片
- 同步
多核并行化
通过了解 CPU 核心数量,并针对性地分解计算任务到多个 goroutine 中并行计算。
type Vector []float64
func (v Vector) DoSome(i, n int, u Vector, c chan int) {
for ; i < n; i++ {
v[i] += u.Op(v[i])
}
c <- 1 // 发信号告诉任务管理者已经计算完成
}
const NCPU = 16
func (v Vector) DoAll(u Vector) {
c := make(chan int, NCPU) // 用于接收每个 CPU 的任务完成信号
for i := 0; i < NCPU; i++ {
go v.DoSome(i*len(v)/NCPU, (i+1)*len(v)/NCPU, u, c)
}
// 等待所有 CPU 的任务完成
for i := 0; i<NCPU; i++ {
<-c // 获取到一个数据,表示一个 CPU 计算完成了
}
// 到这里表示所有计算已经结束
}
目前还需要通过设置环境变量 GOMAXPROCS 的值来控制使用多少个 CPU 核心。
// 在代码启动 goroutine 之前调用
runtime.GOMAXPROCS(16)
可以通过 runtime.NumCPU() 来获取核心数。
出让时间片
这涉及到比较精细的控制 goroutine 的行为,需要比较深入地了解 Go 中的 runtime 包提供的具体功能。
同步
同步锁
除了使用 channel ,Go 也提供资源锁的方案。
包括 2 种类型的锁:
- sync.Mutex
- sync.RWMutex
当一个 goroutine 获得了 Mutex 之后,其他的 goroutine 就只能等待这个 goroutine 释放此 Mutex。
RWMutex 是经典的单写多读模型。在读锁占用的情况下,会阻止写,但不阻止读;在写锁占用时,会阻止所有的读写。
从数据结构上来看,RWMutex 组合了 Mutex:
type RWMutex struct {
w Mutex
writerSem uint32
readerSem uint32
readerCount int32
readerWait int32
}
锁的基本使用是,要保证上锁和解锁的对应。标准的锁用法如下:
var l sync.Mutex
func foo() {
l.Lock()
defer l.Unlock()
}
全局唯一操作
对于从全局角度只需要运行一次的代码:
- 全局初始化
Go 中提供 Once 类型来保证全局的唯一操作。
var a string
var once sync.Once
func setup() {
a = "hello, world"
}
func doprint() {
once.Do(setup)
print(a)
}
func twoprint() {
go doprint()
go doprint()
}
- 上一篇: go并发编程入门-基础术语
- 下一篇: Go 语言 Goroutines 协程并发
猜你喜欢
- 2024-12-01 Go 并发可视化解释 — 通道
- 2024-12-01 Go并发编程面试15题
- 2024-12-01 Go 语言并发编程实践:从入门到进阶
- 2024-12-01 Go语言编写的简单并发编程示例
- 2024-12-01 并发编程的奇技淫巧:Go语言调度器的内在工作机制
- 2024-12-01 Goroutine 并发调度模型深度解析之手撸一个高性能 goroutine 池
- 2024-12-01 每天2分钟学习GO语言编程(十九)并发简明教程
- 2024-12-01 并发编程,程序员必修课,来看go语言的巧妙实现,极为干练
- 2024-12-01 Go项目中如何限制并发数?Atomic必须掌握
- 2024-12-01 Go语言并发入门
- 1509℃桌面软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具
- 526℃Dify工具使用全场景:dify-sandbox沙盒的原理(源码篇·第2期)
- 492℃MySQL service启动脚本浅析(r12笔记第59天)
- 472℃服务器异常重启,导致mysql启动失败,问题解决过程记录
- 469℃启用MySQL查询缓存(mysql8.0查询缓存)
- 449℃「赵强老师」MySQL的闪回(赵强iso是哪个大学毕业的)
- 429℃mysql服务怎么启动和关闭?(mysql服务怎么启动和关闭)
- 426℃MySQL server PID file could not be found!失败
- 最近发表
- 标签列表
-
- c++中::是什么意思 (83)
- 标签用于 (65)
- 主键只能有一个吗 (66)
- c#console.writeline不显示 (75)
- pythoncase语句 (81)
- es6includes (73)
- windowsscripthost (67)
- apt-getinstall-y (86)
- node_modules怎么生成 (76)
- chromepost (65)
- c++int转char (75)
- static函数和普通函数 (76)
- el-date-picker开始日期早于结束日期 (70)
- js判断是否是json字符串 (67)
- checkout-b (67)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- & (66)
- java (73)
- js数组插入 (83)
- linux删除一个文件夹 (65)
- mac安装java (72)
- eacces (67)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)