这篇文章主要讲解了“怎么使用Golang中的select语句实现并发编程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么使用Golang中的select语句实现并发编程”吧!
序文
select 是用来配合channel使用的
空select
没有内容的select 会阻塞
没有内容是指,没有case,也没有default
如果没有其它的任务指执行,将会触发死锁
package main
import (
    "fmt"
    "time"
)
/**
* 没有内容的select 会阻塞
* 没有内容是指,没有case,也没有default
* 如果没有其它的任务指执行,将会触发死锁
*/
func main() {
    go func() {
        time.Sleep(2 * time.Second)
        fmt.Println("没有内容的select 会阻塞")
    }()
    select {}
}只有default的select
只有default的select 和串行化没有区别
package main
import "fmt"
/**
 只有default的select 和串行化没有区别
 */
func main() {
    go func() {
        fmt.Println("quick")
    }()
    select {
    default:
        fmt.Println("end")
    }
}带 case 的 select
有case,有 default
如果能匹配到case 就 执行 case
匹配不到case,就执行default
有 default,就代表了不会阻塞
package main
import (
    "fmt"
)
func main() {
    ch2 := make(chan int, 2)
    ch3 := make(chan int, 2)
    select {
    case v1 := <-ch2:
        fmt.Println(v1)
    case v2 := <-ch3:
        fmt.Println(v2)
    default:
        fmt.Println(22)
    }
}package main
import (
    "fmt"
    "time"
)
func main() {
    ch2 := make(chan int, 2)
    ch3 := make(chan int, 2)
    go func() {
        ch2 <- 1
    }()
    time.Sleep(1 * time.Second)
    select {
    case v1 := <-ch2:
        fmt.Printf("get v1 chan value %d", v1)
    case v2 := <-ch3:
        fmt.Printf("get v1 chan value %d", v2)
    default:
        fmt.Println(22)
    }
}有case,无default
会阻塞 一直等到case匹配上
package main
import (
    "fmt"
    "time"
)
func main() {
    ch2 := make(chan int, 2)
    ch3 := make(chan int, 2)
    fmt.Printf("start unix:%d 
", time.Now().Unix())
    go func() {
        time.Sleep(3 * time.Second)
        ch2 <- 1
    }()
    select {
    case v1 := <-ch2:
        fmt.Printf("case unix: %d 
", time.Now().Unix())
        fmt.Printf("get v1 chan value %d 
", v1)
    case v2 := <-ch3:
        fmt.Printf("get v1 chan value %d 
", v2)
    }
    fmt.Println("end")
}select 只匹配一次,如果要进行 n > 1 的 匹配,使用 for + select
package main
import (
    "fmt"
    "time"
)
func main() {
    ch2 := make(chan int, 2)
    ch3 := make(chan int, 2)
    fmt.Printf("start unix:%d 
", time.Now().Unix())
    go func() {
        for {
            time.Sleep(1 * time.Second)
            ch2 <- 1
        }
    }()
    for {
        select {
        case v1 := <-ch2:
            fmt.Printf("case unix: %d 
", time.Now().Unix())
            fmt.Printf("get v1 chan value %d 
", v1)
        case v2 := <-ch3:
            fmt.Printf("get v1 chan value %d 
", v2)
        }
    }
}匹配是无序的
package main
import (
    "fmt"
    "time"
)
func test() {
    ch2 := make(chan int)
    ch3 := make(chan int)
    go func() {
        ch2 <- 1
        close(ch2)
    }()
    go func() {
        time.Sleep(1 * time.Second)
        ch3 <- 1
        close(ch3)
    }()
    time.Sleep(2 * time.Second)
    // 如果有顺序,那么因该每次都是v1
    select {
    case v1 := <-ch2:
        fmt.Printf("case unix: %d 
", time.Now().Unix())
        fmt.Printf("get v1 chan value %d 
", v1)
    case v2 := <-ch3:
        fmt.Printf("case unix: %d 
", time.Now().Unix())
        fmt.Printf("get v2 chan value %d 
", v2)
    }
}
func main() {
    for i := 0; i < 10; i++ {
        test()
    }
}