GoLang 关闭(退出) goroutine 的方式_golang 外部退出go协程

文章目录

GoLang 关闭 goroutine 的方式

一、goroutine介绍

goroutine 是 Go 语言实现并发编程的利器,是 Go 语言中的轻量级线程实现,是非常轻量级的,它就是一段代码,一个函数入口,以及在堆上为其分配的一个堆栈。

在64位机器上,一个 goroutine 初始化大小为2KB,最大为1GB,会随着程序的执行自动增长删除。

goroutine 由 Go 运行时(runtime)管理,简单的一个指令 go func 就能启动一个 goroutine 。但是,Go语言并没有提供终止 goroutine 的接口,也就是说,我们不能从外部去停止一个 goroutine,只能由 goroutine 内部退出(main函数终止除外)。

二、为什么要中断goroutine?

场景:
  2个相互依赖的的操作,“依赖”是指如果其中一个失败,那么另一个就没有意义,而不是第二个操作依赖第一个操作的结果(该情况下,两个操作不能并行)。在这种情况下,如果我们很早就知道其中一个操作失败,那么我们就会希望能取消所有相关的操作。

三、退出goroutine的方法

1、向退出通道发送退出信号

退出一个 goroutine。

select监听多个channel,由于这个特性,for-select结构在Go并发编程中使用的频率很高。

func main() {
	// 定义退出通道
	ch := make(chan bool)
	go func() {
		for {
			select {
			case <-ch:
				fmt.Println("收到退出信号,退出协程!!!")
				return
			default:
				fmt.Println("监控中!!!")
				time.Sleep(1 \* time.Second)
			}
		}
	}()
	time.Sleep(3 \* time.Second)
	// 向退出通道发送退出信号
	ch <- true
	fmt.Println("退出主函数!!!")
}

在这里插入图片描述

2、关闭退出通道

退出多个 goroutine。

当 goroutine 过多时,我们就需要向退出通道中发送多次退出信号,如果再用发送信号的方式就很麻烦,有一个很简单的方法,关闭 channel,这样所有监听退出 channel 的 goroutine 就都会收到关闭信号。

func main() {
	// 定义退出通道
	ch := make(chan bool)
	go func() {
		for {
			select {
			case <-ch:
				fmt.Println("收到退出信号01,退出协程01!!!")
				return
			default:
				fmt.Println("监控01!!!")
				time.Sleep(1 \* time.Second)
			}
		}
	}()
	go func() {
		for {
			select {
			case <-ch:
				fmt.Println("收到退出信号02,退出协程02!!!")
				return
			default:
				fmt.Println("监控02!!!")
				time.Sleep(1 \* time.Second)
			}
		}
	}()
	time.Sleep(3 \* time.Second)
	// 关闭退出通道
	close(ch)
	time.Sleep(5 \* time.Second)
	fmt.Println("退出主函数!!!")
}

在这里插入图片描述

3、使用context.WithCancel() 方法

使用 context.WithCancel() 方法,手动调用 cancel() 方法退出 goroutine。

只有创建它的函数才能调用取消函数来取消此 context。如果你愿意,可以传递取消函数,但是,强烈建议不要这样做,这可能导致取消函数的调用者没有意识到取消 context 的下游影响,可能存在源自此的其他 context,可能导致程序以意外的方式运行。简而言之,永远不要传递取消函数。

func main() {
	// 新建一个上下文
	ctx := context.Background()
	// 在初始上下文的基础上创建一个有取消功能的上下文,需要取消时,就调用cancel(),发出取消信号
	ctx, cancel := context.WithCancel(ctx)
	go func() {
		for {
			select {
			case <-ctx.Done():
				fmt.Println("退出协程!!!")
				return
			default:
				fmt.Println("监控中 WithCancel !!!")
				time.Sleep(1 \* time.Second)
			}
		}
	}()

	time.Sleep(5 \* time.Second)
	// 取消信号
	cancel()
	time.Sleep(2 \* time.Second)
	fmt.Println("退出主函数!!!")
}

在这里插入图片描述

4、使用context.WithTimeout() 方法

使用context.WithTimeout() 方法,手动调用 cancel()方法,在超时之前退出 goroutine。

func main() {
	ctx, cancel := context.WithTimeout(context.Background(), time.Second\*10)
	go func() {
		for {
			select {
			case <-ctx.Done():
				fmt.Println("退出协程!!!")
				return
			default:
				fmt.Println("监控中 WithTimeout !!!")
				time.Sleep(1 \* time.Second)
			}
		}
	}()
	time.Sleep(5 \* time.Second)
	cancel()
	time.Sleep(2 \* time.Second)
	fmt.Println("退出主函数!!!")
}

在这里插入图片描述

5、使用context.WithDeadLine()方法

使用 context.WithDeadLine() 方法,在指定的时间退出 goroutine。

func main() {
	nowTime := time.Now().Add(3 \* time.Second)
	ctx, \_ := context.WithDeadline(context.Background(), nowTime)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值