1.通过代码分析golang语言的执行流程
代码A:
func main() {
//并发执行下面这个方法
go println("Go! Goroutine!")
/* 1.函数time.Sleep()的作用是让调用它的Goroutine暂停(进入Gwaiting状态)一段时间 这里让main函数所在的Goroutine暂停了1毫秒 2.在理想情况下,运行该源码文件会如我们所愿地在标准输出上打印出Go!Goroutine 3.情况并不是总是这样的,调度器的实时调度是我们无法控制的,所以上面这个策略是非常不保险的 4.Go语言还有一个函数runtime.Gosched()替换对time.Sleep()函数的调用是一种更保险的方式 5.runtime.Gosched()函数的作用是让其他Goroutine有机会被运行,但是在实际情况下这个方法也无法保证 */
time.Sleep(time.Millisecond)
}
上面的代码可能会输出Go! Goroutine!,可能什么都没输出。但是测试很难发现没输出这个打印的。
代码B:
func main() {
//定义一个变量并初始化为ABCD
name := "ABCD"
//启动一个协程去处理里面的方法,把它丢入到执行队列中
go func() {
fmt.Printf("Hello,%s.\n",name)
}()
//继续修改name的值
name = "XYZ"
//调用下面的方法让上面的协程方法能够得到执行,但是运行了几次没有打印输出内容,这个跟上面的调度机制是一样的。
runtime.Gosched()
//调用下面函数让主Goroutine进入Gwaiting状态,然后让前面的协程去执行,测试发现加了这延迟,协程一般会得到执行,但是不能保证。
time.Sleep(time.Millisecond*5)
代码C:
func main() {
names :=[]string{"A","B","C","D","E","F"}
for _,name := range names{
go func() {
fmt.Printf("Hello,name)
}()
}
runtime.Gosched()
//第一次执行的结果如下
Hello,F.
Hello,B.
Hello,E.
Hello,F.
//第二次输只有下面几行,第三次一个也没输出
Hello,F.
//从代码执行的结果来分析: 1.6个协程的执行可能全部完成,可能一个也没有执行,这个跟上面的调度策略有关 2.第一次执行结果name的值在不同的协程中有可能是一样的,这个跟name的作用域有关,6个协程使用外部的同一个变量name,所以每个协程被丢入任务队列的时候,可能还没运行,下一次迭代有开始了,然后name的值变成了第一个,或者其他的,然后马上处于Gwaiting状态的协程就马上去执行,这个时候name的执行是目前迭代的那个值,从而可能导致多个协程使用同一个name的值。