某网站的面试题:
首先执行了一下,确实死锁,如图:
按题目要求添加代码:
第一次执行:
第二次:
第三次:
第六次:
因为线程执行顺序并不一定,所以加上答案中的方法,并不一定得到相应的结果。
当然题目要求的是不报错,但是如果写入c2的线程执行晚于读取线程是否还是会报错?
第二次执行已经证明晚于main线程。
改造后的代码:
package main
import (
"fmt"
"runtime"
"time"
)
var c1 = make(chan int)
var c2 = make(chan int)
func main() {
//设置cpu核数为1
runtime.GOMAXPROCS(1)
//让当前线程让出 cpu 以让其它线程运行,它不会挂起当前线程,因此当前线程未来会继续执行
runtime.Gosched()
go func(){
c2<-1
}()
go func() {
fmt.Println("1111111")
runtime.Gosched()
c1 <-<- c2
}()
go func() {
runtime.Gosched()
c1 <-<- c2
fmt.Println("2222222")
}()
<- c1
time.Sleep(3 * time.Second)
}
可以保证输出结果永远是"1111111" "2222222"
runtime:
runtimeruntime
cpuCPUCPUgoroutinedefer
刚开始看到符号"<-<-",百度了一下,查到的都是"chan <-"或者"<- chan",查了好久。直到看了这篇博客:
意思就是将chan中的数据读取出来,放到了另一个chan中。省略了中间变量的写法。
文中还有一点:
查了一下:
官方的go编译器限制channel最多能容纳到65535个元素,尽管如此,我们也不应该传递体积过大的元素值,因为channel的数据从进入到流出会涉及到数据拷贝操作。如果元素体积过大,最好的方法还是使用传递指针来取代传递值。
channel类型是可以带有方向的,假设T是一种类型
chan T是双向channel类型,编译器允许对双向channel同时进行发送和接收。
chan<- T是只写channel类型,编译器只允许往channel里面发送数据。
<-chan T是只读channel类型,编辑器只允许从channel里面接收数据。
双向类型的channel,可以被强制转换成只读channel或者是只写channel,但是反过来却不行,只读和只写channel是不可以转换成双向channel的。
单向channel一般用于方法的入参或返回值。