-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathprimes.go
46 lines (41 loc) · 1.07 KB
/
primes.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
package primes
import (
"math"
)
// Odds writes a sequence of odd numbers starting with `start` to the channel `ch`
func Odds(start uint, ch chan<- uint) {
for i := start; ; i += 2 {
ch <- i // Send 'i' to channel 'ch'.
}
}
// Filter copies the values from channel 'in' to channel 'out',
// removing those divisible by 'prime'.
func Filter(in <-chan uint, out chan<- uint, prime uint) {
for {
i := <-in // Receive value from 'in'.
if i%prime != 0 {
out <- i // Send 'i' to 'out'.
}
}
}
// Nth computes the n-th prime using a daisy-chain filter process, which
// filters composite numbers concurrently
func Nth(n uint) uint {
// Create a new channel and write 2, the only even prime number
ch := make(chan uint, 10)
ch <- 2
// Launch Odds goroutine
go Odds(3, ch)
var prime uint
for i := uint(0); i < n; i++ {
// Read a prime number from `ch`
prime = <-ch
if prime > 2 && float64(i) < math.Sqrt(float64(n)) {
// Filter to ensure that `ch` always returns prime numbers
ch1 := make(chan uint, 100)
go Filter(ch, ch1, prime)
ch = ch1
}
}
return prime
}