Go素数筛选分析详解
来源:脚本之家
时间:2022-12-31 17:27:22 283浏览 收藏
对于一个Golang开发者来说,牢固扎实的基础是十分重要的,golang学习网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《Go素数筛选分析详解》,主要介绍了素数、筛选,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!
Go素数筛选分析
1. 素数筛选介绍
学习Go
语言的过程中,遇到素数筛选的问题。这是一个经典的并发编程问题,是某大佬的代码,短短几行代码就实现了素数筛选。但是自己看完原理和代码后一脸懵逼(仅此几行能实现素数筛选),然后在网上查询相关资料,依旧似懂非懂。经过1天的分析调试,目前基本上掌握了的原理。在这里介绍一下学习理解的过程。
素数筛选基本原理如下图:
就原理来说还是比较简单的,首先生成从 2
开始的递增自然数,然后依次对生成的第 1, 2, 3, ...个素数
整除,经过全部整除仍有余数的自然数,将会是素数。
大佬的代码如下:
// 返回生成自然数序列的管道: 2, 3, 4, ... // GenerateNatural 函数内部启动一个 Goroutine 生产序列,返回对应的管道 func GenerateNatural() chan int { ch := make(chan int) go func() { for i := 2; ; i++ { ch
main()
函数先是调用GenerateNatural()
生成最原始的从2
开始的自然数序列。然后开始一个100
次迭代的循环,希望生成100
个素数。在每次循环迭代开始的时候,管道中的第一个数必定是素数,我们先读取并打印这个素数。然后基于管道中剩余的数列,并以当前取出的素数为筛子过滤后面的素数。不同的素数筛子对应的管道是串联在一起的。运行代码,程序正确输出如下:
1: 2
2: 3
3: 5
......
......
98: 521
99: 523
100: 5412. 代码分析
之前在课本中学习到:
chan底层结构 是一个指针,所以我们能在函数间直接传递 channel,而不用传递 channel 的指针
。上述代码
fun GenerateNatural()
中创建了管道ch := make(chan int)
,并创建一个协程(为了便于描述,该协程称为Gen
)持续向ch
中写入渐增自然数。当
i=0
时,main()
中prime := 读取该
ch
(此时prime=2
,输出素数2
),接着将ch
传入PrimeFilter(ch, prime)
中。PrimeFilter(ch, prime)
创建新协程(称为PF(ch, 2)
)持续读取传入的ch
(ch
中2
之前已被取出,从3
依次往后读取),同时返回一个新的chan out
(当通过过滤器的i
向out
写入时,此时out
仅有写入而没有读取操作,PF(ch, 2)
将阻塞在第1
次写chan out
操作)。与此同时main()
中ch = PrimeFilter(ch, 2)
将out
赋值给ch
,此操作给ch
赋了新变量。到这里,重点来了:由于在随后的时间里,协程Gen
、PF(ch, 2)
中仍需要不停写入和读取ch
,这里将out
赋值给ch
的操作是否会更改Gen
、PF(ch, 2)
两协程中ch
的值了?直接给出答案(后面会给出代码测试),此时
ch
赋新值不影响Gen
、PF(ch, 2)
两协程,仅影响main() for
循环体随后对chan
的操作。(本人认为go
中channel
参数传递采用了channel
指针的拷贝,后续给channel
赋新值相当于将该channel
重新指向了另外一个地址,该channel
与之前协程中使用的channel
分别指向不同地址,是完全不同的变量)。为了便于后面分析,这里将ch = PrimeFilter(ch, 2)
赋值后的ch
称为ch_2
。当
i=1
时,main() for
循环读取前一次产生新的ch_2
赋值给prime
(此时prime=3
,输出素数3
),接着将ch_2
传入PrimeFilter(ch, prime)
并创建新协程(称为PF(ch, 3)
),而后ch = PrimeFilter(ch, 3)
将新产生的out
赋值给ch
,称为ch_3
。与此同时协程Gen
持续向ch
中写入直至阻塞,携程PF(ch, 2)
持续读取ch
值并写入ch_2
直至阻塞,新协程PF(ch, 3)
持续读取ch_2
值并输出至chan out(即ch_3)
(此时ch_3
仅有写入而没有读取操作,PF(ch, 3)
将阻塞在第1
次写ch_3
操作)。当
i
继续增加时,后面的结果以此类推。总结一下:
main()
函数中,每循环1
次,会增加一个协程PF(ch, prime)
,且协程Gen
与新增加的协程之间是串联的关系(即前一个协程的输出,作为下一个协程的输入,二者通过channel
交互),协程main
每次循环读取最后一个channel
的第1
个值,获取prime
素数。基本原理如下图所示。3. 代码验证
(1) channel参数传递验证
func main() { ch1 := make(chan int) go write(ch1) go read(ch1) time.Sleep(time.Second * 3) fmt.Println("main() 1", ch1) ch2 = make(chan int) ch1 = ch2 fmt.Println("main() 2", ch1) time.Sleep(time.Second * 3) } func read(ch1 chan int) { for { time.Sleep(time.Second) fmt.Println("read",测试代码比较简单,在
main()
中创建chan ch1
,后创建两个协程write
、read
分别对ch1
不间断写入与读取,持续一段时间后,main()
新创建ch2
,并赋值给ch1
,查看协程write
、read
是否受到影响。... write 0xc000048120 read 5 0xc000048120 main() 1 0xc000048120 main() 2 0xc000112000 write 0xc000048120 read 5 0xc000048120 ...程序输出如上,可以看到
ch1
地址为0xc000048120
,ch2
地址为0xc000112000
。main()
中ch1
的重新赋值不会影响到其他协程对ch1
的读写。(2) 素数筛选代码验证
在之前素数筛选源码的基础上,添加一些调试打印代码,以便更容易分析代码,如下所示。
package main import ( "fmt" "runtime" "sync/atomic" ) var total uint32 // 返回生成自然数序列的管道: 2, 3, 4, ... func GenerateNatural() chan int { ch := make(chan int) go func() { goRoutineId := atomic.AddUint32(&total, 1) for i := 2; ; i++ { //fmt.Println("before generate", i) ch1)打印协程
id
由于
Go
语言没有直接把获取go
程id
的接口暴露出来,这里采用atomic.AddUint32
原子操作,每次新建1
个协程时,将atomic.AddUint32(&total, 1)
的值保存下来,作为该协程的唯一id
。2)输出结果分析
[routineId: 0002]----generate i=2, ch=0xc000018180
[routineId: 0001]----main i=1; prime=2, ch=0xc000018180, total=2
[routineId: 0003]----read i=3, in=0xc000018180, out=0xc000090000
[routineId: 0002]----generate i=3, ch=0xc000018180
[routineId: 0001]----main i=2; prime=3, ch=0xc000090000, total=3
[routineId: 0002]----generate i=4, ch=0xc000018180
[routineId: 0002]----generate i=5, ch=0xc000018180
[routineId: 0003]----read i=5, in=0xc000018180, out=0xc000090000
[routineId: 0002]----generate i=6, ch=0xc000018180
[routineId: 0002]----generate i=7, ch=0xc000018180
......输出结果如上,
main
协程id=1
,GenerateNatural
协程id=2
,PrimeFilter(ch, prime)
协程id
从3
开始递增。这里还是不太容易看明白,下面分类阐述输出结果。首先,单独查看
GenerateNatural
协程输出,如下。可以看出,此协程就是在写入阻塞交替间往ch=0xc000018180
中写入数据。[routineId: 0002]----generate i=2, ch=0xc000018180 [routineId: 0002]----generate i=3, ch=0xc000018180 [routineId: 0002]----generate i=4, ch=0xc000018180 [routineId: 0002]----generate i=5, ch=0xc000018180 [routineId: 0002]----generate i=6, ch=0xc000018180 [routineId: 0002]----generate i=7, ch=0xc000018180 [routineId: 0002]----generate i=8, ch=0xc000018180 [routineId: 0002]----generate i=9, ch=0xc000018180 ......接着,查看
PrimeFilter(ch, prime)
协程,如下。每输出1
个素数,将增加1
个PrimeFilter(ch, prime)
协程,且协程id
号从3
开始递增。[routineId: 0003]----read i=3, in=0xc000018180, out=0xc000090000 ...... [routineId: 0004]----read i=5, in=0xc000090000, out=0xc0000181e0 ...... [routineId: 0005]----read i=7, in=0xc0000181e0, out=0xc00020a000 ...... [routineId: 0006]----read i=11, in=0xc00020a000, out=0xc00020a060 ......可以看出,协程
[routineId: 0003]
读取GenerateNatural
协程ch=0xc000018180
值作为输入,并将out=0xc000090000
输出作为[routineId: 0004]
协程输入。以此类推,从id>=2
开始的多个协程是通过channel
管道串联在一起的,且前一个协程的输出作为后一个协程的输入。与前述分析一致。最后,查看
main
线程,其id=1
,可见main
每次循环读取最后一个channel
的第1
个值,且该值为素数。与前述分析一致。[routineId: 0002]----generate i=2, ch=0xc000018180 [routineId: 0001]----main i=1; prime=2, ch=0xc000018180, total=2 [routineId: 0003]----read i=3, in=0xc000018180, out=0xc000090000 ...... [routineId: 0001]----main i=2; prime=3, ch=0xc000090000, total=3 ...... [routineId: 0004]----read i=5, in=0xc000090000, out=0xc0000181e0 ...... [routineId: 0001]----main i=3; prime=5, ch=0xc0000181e0, total=4 [routineId: 0005]----read i=7, in=0xc0000181e0, out=0xc00020a000 [routineId: 0001]----main i=4; prime=7, ch=0xc00020a000, total=54. 总结
- 对
Go
不同协程中chan
的传递原理了解不深,且素数筛选代码中多个协程统一使用了ch
名称,特别是对于main()中ch的重新赋值会不会影响其他协程
不甚了解,导致理解混乱。 - 经深入分析代码后理解了素数筛选的内部原理,可谓知其所以然,然如果让自己来设计,代码肯定会臃肿非常多,对于大佬能用如此简单的代码实现功能,万分钦佩!
好了,本文到此结束,带大家了解了《Go素数筛选分析详解》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!
-
190 收藏
-
360 收藏
-
438 收藏
-
280 收藏
-
181 收藏
-
371 收藏
-
236 收藏
-
416 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习