当前位置: 首页 > article >正文

【Golang】协程

  好久没学习golang了,今天学习了一下协程。

文章目录

  • go关键字
  • 通道
  • 等待
  • 互斥锁
  • 选择器

go关键字

使用go关键字,再加一个函数名,就可以开启一个新的协程.

package main

import (
	"fmt"
	"time"
)

func main() {
	go func() {
		fmt.Println("goroutine")
	}()
	fmt.Println("main")
	time.Sleep(time.Second)
}

通道

  通道是协程之间的通讯机制。通过<-运算符向通道写入或读取数据。
创建通道是make函数。从通道读数据时,如果通道没有数据,就会阻塞。
以下是一个简单的例子:

package main

import (
	"fmt"
	"time"
)

func worker(i int, ch chan int) {
	for {
		fmt.Println("worker", i, "通道数据", <-ch)
	}
}
func main() {
	// RoutineDemo()
	ch := make(chan int)
	// 开始3个协程
	for i := 0; i < 3; i++ {
		go worker(i, ch)
	}
	// 向通道写入数据10次
	for i := 0; i < 10; i++ {
		ch <- i
		time.Sleep(time.Second)
	}
}

等待

  在Go语言中,可以使用WaitGroup等待一组协程执行完毕。
前面的例子里,用的是time.Sleep函数,让主协程等待子协程执行完毕。这种写法肯定是不行的,因为很难准确估计子协程执行需要的时间。
幸好go语言提供了WaitGroup,可以很方便地等待所有子协程执行完毕。

package main

import (
	"fmt"
	"math/rand/v2"
	"sync"
	"time"
)

func work(i int, wg *sync.WaitGroup) {
	var n = time.Duration(rand.IntN(3))
	defer wg.Done()
	time.Sleep(time.Second * n)
	fmt.Println(i, "Hello World")
}

func main() {
	wait := sync.WaitGroup{}
	for i := 0; i < 5; i++ {
		go work(i, &wait)
		wait.Add(1)
	}
	wait.Wait()
	fmt.Println("All Done")
}

互斥锁

  在Go语言中,可以使用sync.Mutex互斥锁来控制对共享资源的访问。
在下面的例子中,如果不适用互斥锁,自增的结果就会不正确。

package main

import (
	"fmt"
	"sync"
)

var counter int

func increment(group *sync.WaitGroup, locker *sync.Mutex) {
	locker.Lock()
	counter++
	group.Done()
	locker.Unlock()
}
func main() {

	var lock sync.Mutex
	var wg sync.WaitGroup
	for i := 0; i < 1000; i++ {
		wg.Add(1)
		go increment(&wg, &lock)
	}

	wg.Wait()
	fmt.Println(counter)
}

  去掉互斥锁,我得到的结果是991,而不是1000。

选择器

  Go语言中的select关键字可以用来监听通道上的数据流动。
当通道上有数据流入时,对应的case会被执行。如果没有数据流入,默认的代码会被执行。
比如说主协程可以监听多个通道,每个子协程向不同的通道写入数据。以下是一个简单的例子:

package main

import (
	"fmt"
	"time"
)

func main() {

	ch1 := make(chan string)
	ch2 := make(chan string)

	go func() {
		for {
			time.Sleep(1 * time.Second)
			ch1 <- "hello"
		}
	}()

	go func() {
		for {
			time.Sleep(2 * time.Second)
			ch2 <- "world"
		}
	}()
	for {
		select {
		case msg1 := <-ch1:
			fmt.Println("msg1", msg1)
		case msg2 := <-ch2:
			fmt.Println("msg2", msg2)
		}
	}
}


http://www.kler.cn/a/406331.html

相关文章:

  • 前端-react(class组件和Hooks)
  • 浅析解析 3D NMS 算法及实现
  • RTL8211F 1000M以太网PHY指示灯
  • Spring cloud 一.Consul服务注册与发现(4)
  • 基于Java Springboot付费自习室管理系统
  • OpenCV 计算图像清晰度
  • 迁移学习理论与应用
  • 力扣--LRC 142.训练计划IV
  • Ubuntu ESP32开发环境搭建
  • 五天SpringCloud计划——DAY2之使用Docker完成项目的部署
  • Excel的图表使用和导出准备
  • [面试]-golang基础面试题总结
  • redis7.x源码分析:(4) ae事件处理器(一)
  • 《Django 5 By Example》阅读笔记:p645-p650
  • SQL注入:理解、防范与最佳实践
  • Ubuntu安装Electron环境
  • 学习electron
  • C#实现blob分析——分别基于OpenCvSharp和Emgu实现
  • 力扣 LeetCode 501. 二叉搜索树中的众数(Day10:二叉树)
  • 【vim】vim怎么从指定行到指定行的行首添加内容
  • 真题-桂城2018年六年级
  • OpenCV与AI深度学习|16个含源码和数据集的计算机视觉实战项目(建议收藏!)
  • HarmonyOS . 沉浸状态栏使用
  • Elasticsearch Windows版的安装及启动
  • 14:00面试,14:08就出来了,问的问题有点变态。。。
  • Unreal从入门到精通之如何绘制用于VR的3DUI交互的手柄射线