管道的注意事项和使用细节
- channel 可以声明为只读,或者只写性质
-- 1. 在默认情况下下,管道是双向 var chan1 chan int //可读可写 -- 2 声明为只写 var chan2 chan<- int chan2 = make(chan int, 3) chan2<- 20 //num := <-chan2 //error fmt.Println("chan2=", chan2) -- 3. 声明为只读 var chan3 <-chan int num2 := <-chan3 //chan3<- 30 //err fmt.Println("num2", num2)
- 案例:
- 使用 select 可以解决从管道取数据的阻塞问题
package main import ( "fmt" "time" ) func main() { //使用select可以解决从管道取数据的阻塞问题 //1.定义一个管道 10个数据int intChan := make(chan int, 10) for i := 0; i < 10; i++ { intChan<- i } //2.定义一个管道 5个数据string stringChan := make(chan string, 5) for i := 0; i < 5; i++ { stringChan <- "hello" + fmt.Sprintf("%d", i) } //传统的方法在遍历管道时,如果不关闭会阻塞而导致 deadlock //问题,在实际开发中,可能我们不好确定什么关闭该管道. //可以使用select 方式可以解决 //label: for { select { //注意: 这里,如果intChan一直没有关闭,不会一直阻塞而deadlock //,会自动到下一个case匹配 case v := <-intChan : fmt.Printf("从intChan读取的数据%d\n", v) time.Sleep(time.Second) case v := <-stringChan : fmt.Printf("从stringChan读取的数据%s\n", v) time.Sleep(time.Second) default : fmt.Printf("都取不到了,不玩了, 可以加入逻辑\n") time.Sleep(time.Second) return //break label } } }
- goroutine 中使用 recover,解决协程中出现 panic,导致程序崩溃问题
- 说明:如果我们起了一个协程,但是这个协程出现了panic,如果我们没有甫获这个panic,就会造成整个程序崩溃,这时我们可以在goroutine中使用ecover来捕获panic,进行处理,这样即使这个协程发生的问题,,但是主线程仍然不受影响,可以继续执行。
package main import ( "fmt" "time" ) //函数 func sayHello() { for i := 0; i < 10; i++ { time.Sleep(time.Second) fmt.Println("hello,world") } } //函数 func test() { //这里我们可以使用defer + recover defer func() { //捕获test抛出的panic if err := recover(); err != nil { fmt.Println("test() 发生错误", err) } }() //定义了一个map var myMap map[int]string myMap[0] = "golang" //error } func main() { go sayHello() go test() for i := 0; i < 10; i++ { fmt.Println("main() ok=", i) time.Sleep(time.Second) } }