-
Notifications
You must be signed in to change notification settings - Fork 0
/
main.go
89 lines (74 loc) · 1.71 KB
/
main.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
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
package main
import (
"errors"
"fmt"
"sync"
"time"
)
// processor
// producer -> consumer -> processor -> terminator (выводит на экран результат, в наеш случае, суммы квадратов входящих наруальных чисел)
// ...
// processor
//
// При возникновении ошибки обработки, требуется отменить все последующие расчеты, и вернуть ошибку
const (
limit = 10000
concurrencySize = 5
)
func main() {
numsChan := producer(limit)
res := make(chan int, limit)
go func() {
if err := consumer(numsChan, res, concurrencySize); err != nil {
fmt.Println(err.Error())
}
close(res)
}()
terminator(res)
}
func producer(limit int) chan int {
out := make(chan int, limit)
for i := 1; i <= limit; i++ {
out <- i
}
close(out)
return out
}
func processor(i int) (int, error) {
if i == 10 {
return 0, errors.New("i hate 5")
}
time.Sleep(5 * time.Second)
return i * i, nil
}
func consumer(numsChan, quadsChan chan int, threadsCount int) error {
errChan := make(chan error, 1)
defer close(errChan)
semaphore := make(chan struct{}, threadsCount)
defer close(semaphore)
wg := sync.WaitGroup{}
for i := range numsChan {
if len(errChan) > 0 {
return <-errChan
}
semaphore <- struct{}{}
wg.Add(1)
go func(x int) {
defer wg.Done()
quad, err := processor(x)
if err != nil {
errChan <- err
return
}
quadsChan <- quad
<-semaphore
}(i)
}
wg.Wait()
return nil
}
func terminator(results chan int) {
for i := range results {
fmt.Println(i)
}
}