当尝试向一个通道发送数据同时处理另一个通道时,Gorilla Websocket 示例会挂起吗?
来源:stackoverflow
时间:2024-04-05 09:54:32 327浏览 收藏
来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习Golang相关编程知识。下面本篇文章就来带大家聊聊《当尝试向一个通道发送数据同时处理另一个通道时,Gorilla Websocket 示例会挂起吗?》,介绍一下,希望对大家的知识积累有所帮助,助力实战开发!
我正在关注 gorilla websocket 库的聊天客户端/服务器示例。
https://github.com/gorilla/websocket/blob/master/examples/chat/hub.go#l36
我尝试修改代码以在新客户端连接时通知其他客户端,如下所示:
for { select { case client := <-h.register: h.clients[client] = true // My addition. Hangs after this (no further register/unregister events are processed): h.broadcast <- []byte("Another client connected!") case client := <-h.unregister: if _, ok := h.clients[client]; ok { delete(h.clients, client) close(client.send) } case message := <-h.broadcast: for client := range h.clients { select { case client.send <- message: default: close(client.send) delete(h.clients, client) } } } } }
我的理解是,在外部 for
循环的下一次迭代中,广播通道应该接收该数据并遵循 message
情况中的逻辑,但它只是挂起。
解决方案
集线器的 broadcast
通道未缓冲。无缓冲通道上的通信等待就绪的发送者和就绪的接收者。 hub goroutine 会阻塞,因为该 goroutine 无法同时准备好发送和接收。
将通道从无缓冲通道更改为缓冲通道并不能解决问题。考虑缓冲区容量为1的情况:
return &hub{ broadcast: make(chan []byte, 1), ... }
按照这个时间表:
1 clienta: client.hub.register <- client 2 clientb: c.hub.broadcast <- message 3 hub: case client := <-h.register: 4 hub: h.broadcast <- []byte("another client connected!")
集线器在#4处阻塞,因为#2处的通道已满。将通道容量增加到两个或更多并不能解决问题,因为任意数量的客户端都可以在另一个客户端注册时广播消息。
要解决此问题,请将广播代码移至一个函数,然后从 select 中的两种情况调用该函数:
// sendAll sends message to all registered clients. // This method must only be called by Hub.run. func (h *Hub) sendAll(message []byte) { for client := range h.clients { select { case client.send <- message: default: close(client.send) delete(h.clients, client) } } } func (h *Hub) run() { for { select { case client := <-h.register: h.clients[client] = true h.sendAll([]byte("Another client connected!")) case client := <-h.unregister: if _, ok := h.clients[client]; ok { delete(h.clients, client) close(client.send) } case message := <-h.broadcast: h.sendAll(message) } } }
你的通道是无缓冲的,这意味着每个读/写都会阻塞,直到另一个 goroutine 在同一通道上执行相反的操作。
当您尝试写入 h.broadcast
时,goroutine 会停止,等待读取器。但是同一个 goroutine 应该充当该通道的读取器,但这种情况永远不会发生,因为 goroutine 被写入阻塞了。从而导致程序死锁。
本篇关于《当尝试向一个通道发送数据同时处理另一个通道时,Gorilla Websocket 示例会挂起吗?》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于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次学习