go 使用锁和chanel按顺序进行协程调度的练习

在这里插入图片描述

package main

import (
	"fmt"
	"sync"
	"sync/atomic"
)

func fish(wg *sync.WaitGroup, counter uint64, fishch, catch chan int) {
	//for {
	//	if counter == 2 {
	//		wg.Done()
	//		return
	//	}
	for i := uint64(0); i < counter; i++ {
		<-fishch
		fmt.Println("fish")
		//atomic.AddUint64(&counter, 1) // 原子操作,悲观锁,这里是值拷贝,不加锁也无妨
		catch <- 1

	}
	wg.Done()
}

func dog(wg *sync.WaitGroup, counter uint64, dogch, fishch chan int) {
	for {
		if counter == 2 {
			wg.Done()
			return
		}
		<-dogch
		fmt.Println("dog")
		counter++
		fishch <- 1
	}
}

func cat(wg *sync.WaitGroup, counter uint64, catch, dogch chan int) {
	for {
		if counter == 2 {
			wg.Done()
			return
		}
		<-catch
		fmt.Println("cat")
		atomic.AddUint64(&counter, 1)
		dogch <- 1
	}
}
func main() {
	var (
		wg         sync.WaitGroup
		catcounter uint64
		//fishcounter uint64
		dogcounter uint64
	)

	dogch := make(chan int, 1)
	fishch := make(chan int, 1)
	catch := make(chan int, 1)
	wg.Add(3)
	catch <- 1 // 第一个协程入口的触发条件,否则死锁
	go cat(&wg, catcounter, catch, dogch)
	go dog(&wg, catcounter, dogch, fishch)
	go fish(&wg, 2, fishch, catch)
	wg.Wait()
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: Go语言的Channel是一种特殊的类型,它可以用来在多个goroutine之间进行通信。Channel提供了一种同步的机制,可以用来在goroutine之间传递消息,控制任务的执行顺序,实现并发任务的协调。Channel是Go语言的一个核心概念,可以用来实现不同goroutine之间的通信和同步。 ### 回答2: Go语言中的channel(通道)是一种用于在不同goroutine之间进行通信和同步的机制。在并发编程中,多个goroutine之间的协调和数据交换是非常重要的,而channel提供了一种简单、安全、高效的方式来实现这种通信。 在Go语言中,我们可以使用内置的make函数来创建一个channel。例如: ch := make(chan int) 这将创建一个用于传输整数类型数据的channel。一个channel是一个先进先出(FIFO)队列,可以在不同的goroutine之间发送和接收数据。发送和接收操作都是阻塞的,这意味着当没有可用的数据发送或接收时,goroutine将会被阻塞,直到有可用的数据进行操作。 发送和接收数据可以使用 <- 操作符来进行。例如: ch <- 10 // 向ch发送数据 x := <- ch // 从ch接收数据,并将其赋值给变量x channel的工作原理是通过在内部使用一个有缓冲或无缓冲的队列来缓冲要发送和接收的数据。有缓冲的channel可以在没有接收方的情况下缓冲一定数量的数据,而无缓冲的channel只有在发送和接收同时准备好时才能进行通信,这样可以确保数据的同步性。 channel不仅可以用于数据的传递,还可以用于实现goroutine之间的同步。通过在channel上进行发送和接收操作,可以确保某个goroutine在另一个goroutine完成特定任务之前不会继续执行。 总的来说,channel是Go语言中用于实现并发编程中goroutine之间通信和同步的重要机制。它提供了一种简单、安全、高效的方式来传递数据和实现同步,使得并发编程变得更加简单和可靠。 ### 回答3: Go语言的channel(通道)是一种特殊的数据结构,用于不同 goroutine(协程)之间的通信和数据交换。在Go语言中,channel是一种并发安全的数据类型,可以安全地在多个 goroutine 之间传递数据。 Channel有以下几个特点: 1. 声明和创建通道:使用`make`函数来创建一个channel,可以指定channel的类型和容量。例如,`ch := make(chan int)`可创建一个整数类型的无缓冲通道。 2. 通信机制:Channel通过发送和接收操作来实现通信。发送操作使用`<-`符号,比如`ch <- value`,将value发送到channel ch中。接收操作使用`<-`符号,比如`value := <- ch`,从channel ch中接收数据,并将其赋值给value。 3. 阻塞和非阻塞:当一个goroutine向一个channel发送数据或从一个channel接收数据时,如果通道已满或者通道为空,发送和接收操作都会阻塞当前的goroutine,直到通道有空间或者有数据。这样保证了通道的同步和保序性。 4. 缓冲和非缓冲:无缓冲通道是指没有容量限制的通道。当一个goroutine发送数据到无缓冲通道时,接收goroutine必须立即接收,否则发送goroutine会阻塞。有缓冲通道是指有容量限制的通道。当一个goroutine发送数据到有缓冲通道时,如果通道未满,则发送操作会立即完成,否则发送goroutine会阻塞。 5. 通道的关闭:通过`close`函数可以关闭一个channel。关闭通道后,不能再向通道发送数据,但仍然可以从通道接收数据。接收操作会返回通道的值和一个表示通道是否关闭的布尔值。 通过使用channel,可以在多个goroutine之间进行安全的数据交换,实现了并发程序的同步和协作。使用channel可以有效地避免共享内存的并发访问问题,并提供了一种简单且直观的并发编程模型。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值