Channel使用技巧

前言

Go协程通常使用channel(通道)通讯从而协调/同步他们的工做。合理利用Go协程和channel能帮助咱们大大提升程序的性能。本文将介绍一些使用channel的场景及技巧golang

场景一,使用channel返回运算结果

计算斐波那契数列,在学习递归时候这是个经典问题。如今咱们不用递归实现,而是用channel返回计算得出的斐波那契数列。 计算前40个斐波那契数列的值,看下效率算法

package main

import (
    "fmt"
    "time"
)
//计算斐波那契数列并写到ch中
func fibonacci(n int, ch chan<- int) {
    first, second := 1, 1
    for i := 0; i < n; i++ {
        ch <- first
        first, second = second, first+second
    }
    close(ch)
}

func main() {
    ch := make(chan int, 40)
    i := 0
    start := time.Now()
    go fibonacci(cap(ch), ch)
    for result := range ch {
        fmt.Printf("fibonacci(%d) is: %d\n", i, result)
        i++
    }
    end := time.Now()
    delta := end.Sub(start)
    fmt.Printf("took the time: %s\n", delta)
}

只花了7ms,效率是递归实现的100倍(主要是算法效率问题)数据库

fibonacci(33) is: 5702887
fibonacci(34) is: 9227465
fibonacci(35) is: 14930352
fibonacci(36) is: 24157817
fibonacci(37) is: 39088169
fibonacci(38) is: 63245986
fibonacci(39) is: 102334155
took the time: 8.0004ms

使用for-range读取channel返回的结果十分便利。当channel关闭且没有数据时,for循环会自动退出,无需主动监测channel是否关闭。close(ch)只针对写数据到channel起做用,意思是close(ch)后,ch中不能再写数据,但不影响从ch中读数据并发

场景二,使用channel获取多个并行方法中的一个结果

假设程序从多个复制的数据库同时读取。只须要接收首先到达的一个答案,Query 函数获取数据库的链接切片并请求。并行请求每个数据库并返回收到的第一个响应:函数

func Query(conns []conn, query string) Result {
    ch := make(chan Result, 1)
    for _, conn := range conns {
        go func(c Conn) {
            select {
            case ch <- c.DoQuery(query):
            }
        }(conn)
    }
    return <- ch
}

场景三,响应超时处理

在调用远程方法的时候,存在超时可能,超时后返回超时提示post

func CallWithTimeOut(timeout time.Duration) (int, error) {
    select {
    case resp := <-Call():
        return resp, nil
    case <-time.After(timeout):
        return -1, errors.New("timeout")
    }
}
 
func Call() <-chan int {
    outCh := make(chan int)
    go func() {
        //调用远程方法
    }()
    return outCh
}

一样能够扩展到channel的读写操做性能

func ReadWithTimeOut(ch <-chan int) (x int, err error) {
    select {
    case x = <-ch:
        return x, nil
    case <-time.After(time.Second):
        return 0, errors.New("read time out")
    }
}
func WriteWithTimeOut(ch chan<- int, x int) (err error) {
    select {
    case ch <- x:
        return nil
    case <-time.After(time.Second):
        return errors.New("read time out")
    }
}

使用<-time.After()超时设置可能引起的内存泄露问题,能够看这篇文章学习

场景四,多任务并发执行和顺序执行

方法A和B同时执行,方法C等待方法A执行完后才能执行,main等待A、B、C执行完才退出ui

package main

import (
    "fmt"
    "time"
)

func B(quit chan<- string) {
    fmt.Println("B crraied out")
    quit <- "B"
}

func A(quit chan<- string, finished chan<- bool) {
    // 模拟耗时任务
    time.Sleep(time.Second * 1)
    fmt.Println("A crraied out")
    finished <- true
    quit <- "A"
}

func C(quit chan<- string, finished <-chan bool) {
    // 在A没有执行完以前,finished获取不到数据,会阻塞
    <-finished
    fmt.Println("C crraied out")
    quit <- "C"
}

func main() {
    finished := make(chan bool)
    defer close(finished)
    quit := make(chan string)
    defer close(quit)

    go A(quit, finished)
    go B(quit)
    go C(quit, finished)

    fmt.Println(<-quit)
    fmt.Println(<-quit)
    fmt.Println(<-quit)
}

正常执行咱们获得如下结果code

B crraied out
B
A crraied out
A
C crraied out
C

注意:最后从quit中读数据不能使用for-range语法,否则程序会出现死锁

for res := range quit {
        fmt.Println(res)
    }
fatal error: all goroutines are asleep - deadlock!

缘由很简单,程序中quit通道没有被close,A、B、C运行完了,Go的主协程在for循环中阻塞了,全部Go协程都阻塞了,进入了死锁状态

场景五,超时后中止Go协程,避免浪费资源(中止调用链)

场景四中,假设A方法挂了或者须要执行很长时间,main协程会等到全部方法执行完才会退出。在实际应用中显然不行,因此要设置超时时间。问题来了,C方法是基于A方法执行完后才执行的,咱们怎样通知C方法退出呢。这里针对普通的Go协程,不是Http请求,有关Http超时问题引发的内存泄露能够看这篇文章
下面咱们修改场景四的代码,让A方法有超时设置,C方法在A方法超时后也退出

package main

import (
    "fmt"
    "time"
)

// B方法
func B(quit chan<- string) {
    fmt.Println("B crraied out")
    quit <- "B"
}

// A方法,有超时限制
func AWithTimeOut(quit chan<- string, finishedA chan<- bool, timeout time.Duration) {
    select {
    case resp := <-A(finishedA):
        quit <- resp
    case <-time.After(timeout):
        quit <- "A timeout"
    }
}

// A须要执行的任务
func A(finishedA chan<- bool) <-chan string {
    respCh := make(chan string)
    go func() {
        // 模拟耗时任务
        // time.Sleep(time.Second * 3)
        fmt.Println("A crraied out")
        finishedA <- true
        respCh <- "A"
    }()
    return respCh
}

// C方法,等待A方法完成后才能执行,一样有超时限制,超时时间和A方法一致
func CWithTimeOut(quit chan<- string, finishedA <-chan bool, timeout time.Duration) {
    select {
    case <-finishedA:
        fmt.Println("C crraied out")
        quit <- "C"
    case <-time.After(timeout):
        fmt.Println("C Exited")
        quit <- "C timeout"
    }
}

func main() {
    finishedA := make(chan bool, 1) //这里必需要是1的缓冲通道,否则超时后会死锁
    defer close(finishedA)
    quit := make(chan string, 3)
    defer close(quit)
    timeout := time.Second * 2

    go AWithTimeOut(quit, finishedA, timeout)
    go B(quit)
    go CWithTimeOut(quit, finishedA, timeout)

    fmt.Println(<-quit)
    fmt.Println(<-quit)
    fmt.Println(<-quit)
    time.Sleep(time.Second * 3) //若是程序未退出的话,A方法执行的任务还会继续运行,由于咱们没办法让A方法停下来
}

运行结果

B crraied out
B
C Exited
C timeout
A timeout
A crraied out

A方法用time.Sleep(time.Second * 3)模拟超时任务,代码最后让main协程休眠,主要为了说明虽然A超时了,但正常状况下它仍是会把任务执行下去的。若是有哪位大侠有什么方法能让它不执行,还请告知!!!

总结

本文介绍了几种场景下channel的使用技巧,但愿能起到抛砖引玉的做用,各位若有其它技巧,欢迎评论,本文会把大家的技巧收纳在其中。感谢!!!

相关文章
相关标签/搜索