扇入、扇出 (判断大值素数)

普通的pipeline stage代码

package main

import (
	"math/rand"
	"time"
	"fmt"
)

func main() {

	rands := func() interface{} {return rand.Intn(500000000)}

	toInt := func(done <-chan interface{},
	value <-chan interface{}) <-chan int {
		randInt := make(chan int)
		go func() {
			defer close(randInt)
			for v := range value {
				select {
				case <-done:
					return
				case randInt <- v.(int):
				}
			}
		}()
		return randInt
	}

	repeatRand:= func(done <-chan interface{},
		fn func() interface{} ) <-chan interface{} {


		intData := make(chan interface{})
		go func() {
			defer close(intData)
			for  {
				select {
				case <-done:
					fmt.Println("111Goroutine 退出")
					return
				case intData<- fn():
				}
			}
		}()
		return intData
	}
	Take := func(done <-chan interface{},
		valueStream <-chan interface{},
		num int) <-chan interface{}{

		takeStream := make(chan interface{})
		go func() {
			defer fmt.Println("222Goroutine 退出")
			defer close(takeStream)
			for i := 0; i < num; i++ {
				select {
				case <-done:
					return
				case takeStream <-<-valueStream:
				}
			}
		}()
		return takeStream
	}

	panduan := func(n int) bool {
		for i:=2;i<n;i++ {
			if n % i == 0 {
				return false
			}
		}
		return true
	}

	primeFinder := func(done <-chan interface{},
	value <-chan int) <-chan interface{} {
		primesTREAM := make(chan interface{})
		go func() {

			for v := range value {
				if panduan(v) {
					select {
					case <-done:
						return
					case primesTREAM<- v:
					}
				}else {
					continue
				}
			}
		}()

		return primesTREAM
	}
	done := make(chan interface{})
	start := time.Now()
	randIntStream := toInt(done,repeatRand(done,rands))
	fmt.Println("Primes: ")
	for prime := range Take(done,primeFinder(done,randIntStream),10) {
	fmt.Println(prime)
	}
	close(done)
	fmt.Printf("Search took:%v ",time.Since(start))
}

扇入扇出代码

package main

import (
	"math/rand"
	"time"
	"fmt"
	"runtime"
	"sync"
)

func main() {

	rands := func() interface{} {return rand.Intn(500000000)}

	toInt := func(done <-chan interface{},
	value <-chan interface{}) <-chan int {
		randInt := make(chan int)
		go func() {
			defer close(randInt)
			for v := range value {
				select {
				case <-done:
					return
				case randInt <- v.(int):
				}
			}
		}()
		return randInt
	}

	repeatRand:= func(done <-chan interface{},
		fn func() interface{} ) <-chan interface{} {


		intData := make(chan interface{})
		go func() {
			defer close(intData)
			for  {
				select {
				case <-done:
					fmt.Println("111Goroutine 退出")
					return
				case intData<- fn():
				}
			}
		}()
		return intData
	}
	Take := func(done <-chan interface{},
		valueStream <-chan interface{},
		num int) <-chan interface{}{

		takeStream := make(chan interface{})
		go func() {
			defer fmt.Println("222Goroutine 退出")
			defer close(takeStream)
			for i := 0; i < num; i++ {
				select {
				case <-done:
					return
				case takeStream <-<-valueStream:
				}
			}
		}()
		return takeStream
	}

	panduan := func(n int) bool {
		for i:=2;i<n;i++ {
			if n % i == 0 {
				return false
			}
		}
		return true
	}

	primeFinder := func(done <-chan interface{},
	value <-chan int) <-chan interface{} {
		primesTREAM := make(chan interface{})
		go func() {

			for v := range value {
				if panduan(v) {
					select {
					case <-done:
						return
					case primesTREAM<- v:
					}
				}else {
					continue
				}
			}
		}()

		return primesTREAM
	}

	fanIn := func(done <- chan interface{},
	channels ...<-chan interface{}) <-chan interface{} {
		var wg sync.WaitGroup

		multiPlexedStream := make(chan interface{})
		mutiplex := func(c <-chan interface{}) {
			defer wg.Done()
			for i:= range c {
				select {
				case <-done:
					return
				case multiPlexedStream<-i:
				}
			}
		}
		wg.Add(len(channels))
		for _,c := range channels {
			go mutiplex(c)
		}
		go func() {
			wg.Wait()
			close(multiPlexedStream)
		}()
		return multiPlexedStream
	}

	done := make(chan interface{})

	start := time.Now()
	randIntStream := toInt(done,repeatRand(done,rands))
	// fmt.Println("Primes: ")
	// for prime := range Take(done,primeFinder(done,randIntStream),10) {
	// 	fmt.Println(prime)
	// }
	numinders := runtime.NumCPU()
	fmt.Printf("Spanning up %d prime finders", numinders)

	finders := make([]<-chan interface{},numinders)
	fmt.Println("Primes: ")
	for i := 0; i < numinders; i++ {
		finders[i] = primeFinder(done,randIntStream)
	}

	for prime := range Take(done,fanIn(done,finders...),10){
		fmt.Printf("\t%v\n",prime)
	}

	close(done)
	fmt.Printf("Search took:%v ",time.Since(start))
}

性能对比
普通:
24s左右
扇入扇出
12s左右
性能提升了50%.

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值