二、Golang Channel通信和控制题目

devtools/2025/2/11 13:46:59/

要求

采用Golang语言中channel实现协程间的通讯,并通过控制channel控制起输入与输出格式。其中一个协程sender负责发送字符串"ABC"的字符,另一个协程reciever负责接收这些字符并打印。
要求接收方reciever协程能够按照顺序,按照行输出对接收的值打印出"ABC"(或"AB",跳过"C"),重复10次。

实现代码

以下下代码实现了sender goroutine和revicever goroutine(主协程g0)通过通讯控制,按照字符串顺序接收并打印出AB"(跳过"C"),实现重复10次。
另外,对于跳过的"C"字符的轮齿,打印" No.X"(X为循环输出行的对应次数,输出行内容对应ABC或者AB)。
计时并打印程序运行的总时间。

package mainimport ("fmt""time"
)func main() {content := "ABC"sig := make(chan bool)ch := make(chan rune, 3)go func() {for i := 0; i < 10; i++ {for _, c := range content {ch <- c}sig <- true}close(ch)}()i, j := 1, 1now := time.Now()/*output: ABCfor v := range ch {if i%3 == 0 {<-sigfmt.Printf("%s No.%d\n", string(v), j)j++} else {fmt.Printf("%s", string(v))}i++}*/// output: ABfor v := range ch {if i%3 > 0 {fmt.Printf("%s", string(v))} else {fmt.Printf(" No.%d\n", j)<-sigj++}i++}close(sig)d := time.Since(now)fmt.Printf("Running time: %v\n", d)
}

题解分析

以上代码涉及了 Go 语言中的协程(goroutine)、通道(channel)和时间测量。我们将逐段分析代码的执行逻辑,特别是协程操作控制部分。

代码结构

1. List item

  • 初始化变量和通道: content 字符串包含 “ABC”。

  • sig 是一个布尔类型的通道,用于信号通知。

  • ch是一个有缓冲区的通道,可以存储 3 个 rune 类型的值。

2. 启动协程

  • 协程中,循环 10 次,每次将 content 中的字符发送到 ch 通道,并在每次循环结束时向 sig 发送一个 true 信号。
  • 循环结束后,关闭 ch 通道。

3.主函数中的 for 循环

  • 使用 for v := range ch 接收 ch 通道中的值。

  • 根据条件打印字符或处理信号,通过sig通道控制输入协程

协程操作控制逻辑

关键逻辑在于 for v := range ch 循环中的条件判断:

for v := range ch {if i%3 > 0 {fmt.Printf("%s", string(v))} else {fmt.Printf(" No.%d\n", j)<-sigj++}i++
}

变量 i 和 j:

  • i 用于计数,从 1 开始递增。

  • j 用于生成信号后的编号,从 1 开始递增。

条件判断:

  • 如果 i%3 > 0,即 i 不是 3 的倍数时,直接打印字符 v。

  • 如果 i 是 3 的倍数时,执行以下操作:

    • 打印 " No.%d\n",其中 %d 被 j 的值替换。
    • 从 sig 通道接收信号(阻塞操作,直到 sig 有信号发送)。
    • j 自增。

通道关闭:

  • 协程在发送完 10 轮字符后关闭 ch 通道,此时 for v := range ch 循环结束。

输出分析

根据代码逻辑,输出会如下进行:

  • 第一轮:打印 “A”, “B”(因为 i 为 1 和 2,都不是 3 的倍数)。

  • 第二轮(i 为 3):打印 " No.1"(因为 i 是 3 的倍数),然后等待 sig 信号(此时协程发送完第一轮信号)。

  • 后续字符继续按此逻辑打印,但注意 sig信号是在每一轮结束后发送的,所以每次 i 为 3 的倍数时,会先打印编号再等待信号。

但是,由于主循环在 i 为 3 的倍数时等待 sig信号,而 sig 信号在每一轮字符发送完后才发送,因此:

  • 在打印完 “AB” 后,i 变为 3,此时会等待第一个 sig 信号(即第一轮结束后发送的信号)。

  • 在接收到信号后,打印 " No.1"并继续。

然而,由于代码中的 for 循环逻辑和信号发送的频率,主循环会在每次 i 为 3 的倍数时等待信号,且 sig信号每轮只发送一次,因此输出将受限于信号发送和接收的同步。

最终结果

实际运行代码时,由于信号同步和通道缓冲区的限制,输出将会是按顺序的 “AB” 后接一个编号,但由于信号在每轮结束后才发送,且主循环在 i 为 3 的倍数时等待信号,实际输出将受限于信号发送的时机和通道缓冲区大小。理论上,如果缓冲区足够且信号同步,输出可能是:

AB No.1
...
AB NO.10

但考虑到信号和缓冲区同步的细节,实际运行可能更复杂,且由于代码逻辑,No.2 及之后的编号可能无法按预期打印,因为主循环在接收到信号后会继续,而信号发送频率与字符发送频率不匹配。

注意

  • 代码中的 close(sig) 是多余的,因为 sig 通道在协程结束时并不会被主循环关闭,且主循环结束后程序也会终止,无需显式关闭 sig。
  • 代码中的时间测量部分用于记录程序运行时间,对逻辑分析无直接影响。

http://www.ppmy.cn/devtools/157944.html

相关文章

8种方法解决电脑QQ能上网但是浏览器不能上网问题

刚接触电脑的时候遇到过很多的问题&#xff0c;每次遇到都让我焦头烂额&#xff0c;特别是一些问题不知道怎么解决&#xff0c;忙活很久还是没进展。当时遇到的问题是电脑QQ能上网但是浏览器打不开网页。明明QQ、视频软件也能正常使用&#xff0c;但是浏览器就是打不开网页&…

【数据安全】现代智能手机的数据加密机制

这两天刷抖音看到一个降智视频&#xff0c;说手机拿去修了明明没给他密码&#xff0c;但是他是怎么导出手机的数据呢&#xff0c;而且很多人看你都会有这样的疑问&#xff1a;“我的手机设有锁屏密码数据真的安全吗&#xff1f;锁屏密码存在意义是不是仅限于防止别人进入桌面这…

DeepSeek做赛车游戏

赛车模型 2D生成图片 任意AI图片软件SD&#xff0c;MJ 图片生成3D模型 车身 车轮 场景 Rodin,Tripo和Meshy 询问deepSeek如何开发 拷贝代码 将汽车运行代码拖到汽车上 再让AI写个摄像头跟随代码 再去提问deepseek控制轮胎和一些处理细节

iOS AES/CBC/CTR加解密以及AES-CMAC

感觉iOS自带的CryptoKit不好用&#xff0c;有个第三方库CryptoSwift还不错&#xff0c;好巧不巧&#xff0c;清理过Xcode缓存后死活下载不下来&#xff0c;当然也可以自己编译个Framework&#xff0c;但是偏偏不想用第三方库了&#xff0c;于是研究了一下&#xff0c;自带的Com…

Vue 过渡动画实现全解析:打造丝滑交互体验

Vue 过渡动画实现全解析&#xff1a;打造丝滑交互体验 在当今竞争激烈的 Web 开发领域&#xff0c;用户体验已成为衡量项目成功与否的关键指标。过渡动画作为提升用户体验的利器&#xff0c;能让应用的交互更加丝滑流畅&#xff0c;给用户带来愉悦的使用感受。在 Vue.js 框架中…

微信小程序地图开发总结-规划路线

在现代移动应用中&#xff0c;地图导航功能已成为必不可少的一部分。通过地图 API&#xff0c;我们可以轻松地在应用中集成位置服务和路径规划功能。本篇文章将带大家一起实现一个简单的路径导航功能&#xff0c;使用腾讯地图 API结合微信小程序&#xff0c;实现从当前位置到目…

机器视觉3D相机分哪些

机器视觉中的3D相机主要分为以下几类&#xff1a; 结构光相机 原理&#xff1a;通过投射特定光图案&#xff08;如条纹或点阵&#xff09;到物体表面&#xff0c;利用摄像头捕捉变形图案&#xff0c;计算深度信息。 优点&#xff1a;精度高&#xff0c;适合静态场景。 缺点&…

服务器重启后报Predis_ServerException: Client sent AUTH, but no password is set

Redis问题产生后,处理办法 2025/02/08 11:21:43 [error] [exception.Predis_ServerException] Predis_ServerException: Client sent AUTH, but no password is set in /www/wwwroot/er/protected/extensions/redis/Predis.php:573 Stack trace: #0 /www/wwwroot/er/protected…