Go 中如何在具有互斥体的结构体上实现互斥范围
来源:stackoverflow
时间:2024-02-26 14:33:27 401浏览 收藏
从现在开始,努力学习吧!本文《Go 中如何在具有互斥体的结构体上实现互斥范围》主要讲解了等等相关知识点,我会在golang学习网中持续更新相关的系列文章,欢迎大家关注并积极留言建议。下面就先一起来看一下本篇正文内容吧,希望能帮到你!
我正在尝试 go 并尝试在服务器中进行并发状态管理的各种方法。假设我们有以下内容:
type resource struct {
data int
}
func (r *resource) increment () {
r.data++
}
type client struct {
id int
resource resource
mu sync.rwmutex
}
type activeclients struct {
clients []client
mu sync.rwmutex
}
func (ac *activeclients) add(client client) {
ac.mu.lock()
defer ac.mu.unlock()
if ac.clients == nil {
ac.clients = make([]client, 0)
}
ac.clients = append(ac.clients, client)
}
activeclients.mu 将用于读取和写入 activeclients.clients 切片,而 client.mu 将用于读取和写入 client.resource。现在假设我们要迭代 activeclients.clients 以更新其中一个资源。以下内容会产生错误:
func (ac *ActiveClients) addToResource(clientId int) {
for _, existingClient := range ac.clients {
if existingClient.id == clientId {
existingClient.Lock()
defer existingClient.Unlock()
existingClient.resource.increment()
}
}
}
这会产生“range varexistingclient复制锁:{modulename}.client包含sync.rwmutex”。
如何在不复制锁的情况下遍历切片?
正确答案
for _, v := range s 语句将 s 的元素分配给局部变量 v。该值被复制。没有引用语义
go vet 命令警告您互斥字段已被复制。互斥体一旦使用就不应被复制。
这并不是 incrementresource 中的唯一问题。该函数修改局部变量中的 client 副本,而不是切片中的 client。由于局部变量在函数返回时被丢弃,因此函数 incrementresource 不起作用。运行程序 https://go.dev/play/p/kl9gzsl6d2j 查看演示的问题。
通过指针访问切片元素,修复incrementresource中的错误。
func (ac *activeclients) addtoresource(clientid int) {
for i := range ac.clients {
existingclient := &ac.clients[i] // existingclient is ptr to slice element
if existingclient.id == clientid {
existingclient.lock()
defer existingclient.unlock()
existingclient.resource.increment()
fmt.println("data in addtoresource: ", existingclient.resource.data)
}
}
}
这是修复后的程序:https://go.dev/play/p/wmsuojotaub
上述更改解决了问题中的问题,但这并不是应用程序的唯一问题。方法 activeclients.add 中对 append 的调用会在增长切片时复制 client 值。此复制会在切片元素上创建数据争用,并违反互斥锁一旦使用就不应复制的规则。
要修复所有问题,请使用 *client 的切片而不是 client。当我们这样做时,利用 append 对 nil 切片的处理。
type ActiveClients struct {
clients []*Client
mu sync.RWMutex
}
func (ac *ActiveClients) add(client *Client) {
ac.mu.Lock()
defer ac.mu.Unlock()
ac.clients = append(ac.clients, client)
}
这是最终的程序:https://go.dev/play/p/mink90zdncu
本篇关于《Go 中如何在具有互斥体的结构体上实现互斥范围》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习