网站首页 > 技术文章 正文
对并发的处理还有 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语言并发入门
- 最近发表
-
- count(*)、count1(1)、count(主键)、count(字段) 哪个更快?
- 深入探索 Spring Boot3 中 MyBatis 的 association 标签用法
- js异步操作 Promise fetch API 带来的网络请求变革—仙盟创梦IDE
- HTTP状态码超详细说明_http 状态码有哪些
- 聊聊跨域的原理与解决方法_跨域解决方案及原理
- 告别懵圈!产品新人的接口文档轻松入门指南
- 在Javaweb中实现发送简单邮件_java web发布
- 优化必备基础:Oracle中常见的三种表连接方式
- Oracle常用工具使用 - AWR_oracle工具有哪些
- 搭载USB 3.1接口:msi 微星 发布 990FXA Gaming 游戏主板
- 标签列表
-
- cmd/c (90)
- c++中::是什么意思 (84)
- 标签用于 (71)
- 主键只能有一个吗 (77)
- c#console.writeline不显示 (95)
- pythoncase语句 (88)
- es6includes (74)
- sqlset (76)
- apt-getinstall-y (100)
- node_modules怎么生成 (87)
- chromepost (71)
- flexdirection (73)
- c++int转char (80)
- mysqlany_value (79)
- static函数和普通函数 (84)
- el-date-picker开始日期早于结束日期 (76)
- js判断是否是json字符串 (75)
- asynccallback (71)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)