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 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习