某网站的面试题:

首先执行了一下,确实死锁,如图:

按题目要求添加代码:

第一次执行:

第二次:

第三次:

第六次:

因为线程执行顺序并不一定,所以加上答案中的方法,并不一定得到相应的结果。

当然题目要求的是不报错,但是如果写入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一般用于方法的入参或返回值。