实现一个交替打印字母和数字的简单程序,呈现的效果是:

a1b2c3d4e5f6g7h8i9j10k11l12m13n14o15p16q17r18s19t20u21v22w23x24y25z26

这个效果其实有很多种写法,现在选取一个简单的写法,通常的写法是让两个协程交替打印,用chan来阻塞通知,但是由于大部分的demo都没有考虑优雅退出的问题,这次的demo添加了一个stopCha来优雅退出并close通道

package main

import (
	"fmt"
	"sync"
)

func main() {
	fmt.Println("开始打印")
	printLettersAndNumbAlternately()
	fmt.Println("打印结束")
}

func printLettersAndNumbAlternately() {
	// letterChan 字母打印通知->字母打印完成后就通知数字打印的协程,并阻塞自己
	// numChan 数字打印通知->阻塞自己通知字母打印协程
	// stopChan 结束通知器->字母打印完成后通知全局退出
	letterChan, numChan, stopChan := make(chan struct{}), make(chan struct{}), make(chan struct{}, 1)
	wait := sync.WaitGroup{}
	wait.Add(3)

	go func() {
		defer wait.Done()
		defer close(numChan)
		num := 1
		for {
			select {
			case <-letterChan:
				fmt.Print(num)
				select {
				case numChan <- struct{}{}:
				case <-stopChan:
					return
				}
				num++
			default:
			}
		}
	}()
	go func() {
		defer wait.Done()
		defer close(letterChan)
		letters := get26Letter()
		point := 0
		for {
			select {
			case <-numChan:
				fmt.Print(letters[point])
				letterChan <- struct{}{}
				point++
				if point == len(letters) {
					select {
					case stopChan <- struct{}{}:
					default:
					}
					return
				}
			default:
			}
		}
	}()
	// 主持人
	go func() {
		defer wait.Done()
		// 先打印字母
		numChan <- struct{}{}
		return
	}()
	wait.Wait()
	fmt.Println("\n所有协程结束")
	// 一切结束后再关闭 stop通知器
	close(stopChan)
}

// 获取26个英文字母
func get26Letter() []string {
	var letters []string
	counter := 0
	for i := 'a'; i <= 'z'; i++ {
		letters = append(letters, string(i))
		counter++
	}
	return letters
}