JavaScript 异步回调函数无法实现 WebSocket 非阻塞行为
来源:stackoverflow
时间:2024-03-06 23:30:27 201浏览 收藏
在Golang实战开发的过程中,我们经常会遇到一些这样那样的问题,然后要卡好半天,等问题解决了才发现原来一些细节知识点还是没有掌握好。今天golang学习网就整理分享《JavaScript 异步回调函数无法实现 WebSocket 非阻塞行为》,聊聊,希望可以帮助到正在努力赚钱的你。
我正在运行两个用 go 编写的进程,其中一个进程 (sender.go) 向另一个进程 (listener.go) 发送消息,同时通过 websockets 陷入 for 循环。
问题是listener.go只有在终止循环后才意识到它收到了消息。
我已经尝试了几个 websocket 库,我什至尝试使用常规的 tcp 流,但在 webassembly 中编译时它不会工作,因为浏览器不支持它。尽管有这种行为,syscall/js websocket 似乎是完美的选择。
这里是listener.go
func registercallbacks(ws js.value) { ws.call("addeventlistener", "message", js.funcof(func(this js.value, args []js.value) interface{} { message := args[0].get("data").string() fmt.println("message received ") fmt.println(message) return nil })) } func main() { c := make(chan struct{}, 0) toip := "127.0.0.1" toport := 8081 ws := js.global().get("websocket").new(fmt.sprintf("ws://%s:%d/ws", toip, toport)) registercallbacks(ws) const bignum uint64 = 100000 * 10000 cb := func(this js.value, args []js.value) interface{} { for i := uint64(0); i < bignum; i++ { dothings() if i%10000000 == 0 { fmt.println("skimmed ten millions ") } } fmt.println("exited for loop !!") return nil } // call cb() in script of index.html from a button js.global().set("cb", js.funcof(cb)) <-c }
sender.go 也是如此,
func main() { toip := "127.0.0.1" toport := 8081 ws := js.global().get("websocket").new(fmt.sprintf("ws://%s:%d/ws", toip, toport)) time.sleep(1 * time.second) fmt.println("sending ") ws.call("send", js.valueof("msg")) fmt.println("program exit ") }
如果有人愿意重现这个问题,这里是 websocket 服务器,它接收来自发送者的消息并将其转发给接收者,用 node.js 编写,只需复制并粘贴它,它可以在多个项目中正常工作
const port = 8081; const host = '127.0.0.1'; var WebSocketServer = require('websocket').server; var http = require('http'); var server = http.createServer(function(request, response) {}); server.listen(port, host, () => { console.log('WS Server is running on port ' + port + '.'); }); wsServer = new WebSocketServer({ httpServer: server }); let sockets = []; wsServer.on('request', function(request) { var connection = request.accept(null, request.origin); sockets.push(connection) console.log("Connection with node initiated") connection.on('message', function(message) { console.log("message received "+message) sockets.forEach(function(s, index, array) { if (message.type === 'utf8') { broadcastData = message.utf8Data console.log("message is: "+ broadcastData) if(s!= connection) { console.log('send data to ' + s.socket.remotePort + ': ' + broadcastData); s.sendUTF(broadcastData) } } }); }); });
我期望listener.go在退出for循环之前接收消息
ps:使用 sleep 语句并不是一个好的帮助,因为当这个 for 循环在 js 回调中运行时,sleep 语句会输出一个panic。
解决方案
首先,请注意,如果您生成 wasm Go 代码,它不是多线程的,至少在今天(2019 年 9 月)不是。看 How to implement multithreading in wasm created using golang? 所以只有一个实际执行线程。
然后,请记住,goroutine 可以在任何可用线程上协作执行多任务。由于只有一个线程,因此任何时候都只有一个 goroutine 在运行。如果你正在运行的 goroutine 没有将处理器交给其他 goroutine,那么其他 goroutine 将不会运行。
你可以隐式放弃处理器——例如,通过等待通道或互斥体——或者显式地,通过 time.Sleep
或 runtime.Gosched
。如果没有这样的调用,任何注意到传入消息的 goroutine 永远没有机会看到它和你的回调(将在这个或某个第三个 goroutine 中运行)永远不会被调用。
(如果您在本机运行,而不是在 wasm 中运行,您通常会获得更多线程,并且能够让其他 CPU 运行其他任务,即使一个 CPU 处于紧密循环中。)
以上就是《JavaScript 异步回调函数无法实现 WebSocket 非阻塞行为》的详细内容,更多关于的资料请关注golang学习网公众号!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
478 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 511次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 498次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习