GO语言-10了解Go并发的使用(上)

简介: GO语言的学习与记录,第十篇了解Go语言并发的使用(上),围绕的使用go关键字开启协程,学习协程之间数据传输,了解阻塞式通道和非阻塞式通道,学习如何取通道的值还有怎么遍历通道。最后两个案例助于理解。
初心是记录和总结,自己学习Go语言的历程。如果能帮助到你,这是我的荣幸。

并发是属于比较难的内容,我还不足以把握到其精髓,只能分享一些基本使用和注意事项。

如何开启并发

很简单,只要在方法前使用go关键字,看示例代码。这里使用了匿名函数,匿名函数可以在方法中直接调用执行,通过{}后加上(),也可以传入参数,只需要func(参数)和最后的括号里加入传入(传入参数)对应即可。

package main

import "fmt"

func main() {
   go func() {
      fmt.Println("并发程序")
   }()
   fmt.Println("主程序")
}

程序运行后,会输出主程序,发现并没有打印并发程序。这是因为在Go中,主程序也算是一个并发的主协程,而主协程执行完毕后,会将其他的子协程(通过go关键字开启的)强制结束。这里出现了协程两个字,这是在go中特有的描述并发的,协程就是通过go关键字开启的,英文名叫:goroutine

主协程和子协程如何协调

这个协调是指:主协程能不能等我自己开启的协程结束后才结束。并发本身就是脱离了程序定义的顺序,它并不按照程序语句的顺序执行,而是通过计算机资源调度,和CPU有关系。那么我们如何进行控制呢?

睡眠

睡眠的方式非常简单粗暴,强制该程序‘睡眠’,暂时停止执行。这样CPU就会执行其他协程了。
使用方法:

time.Sleep(d Duration)

Duration是指时间间隔,常用的有

  • time.Hour 一小时
  • time.Minute 一分
  • time.Second 一秒

使用WaitGroup

WaitGroup等待goroutine的集合完成。主程序调用Add来设置等待的gor例行程序的数量。然后每个goroutine运行并在完成时调用Done。同时,Wait可用于阻塞直到所有goroutine完成。

通过这段话我们可以捕获三个点

  • 在主程序中调用WaitGroup的Add方法添加字子协程的数量
  • 每个子协程运行完成时调用Done方法
  • 主程序通过使用Wait方法等待子协程运行结束
// 定义WaitGroup对象
var wg sync.WaitGroup

func main() {
   wg.Add(1) // 添加一个子协程
   go func() {
      fmt.Println("并发程序")
      wg.Done() // 子协程结束后调用Done
   }()
   fmt.Println("主程序")
   wg.Wait() // 等待子协程运行结束,主协程才结束
}

数据传递 - 通道

并发执行时第一个重要的点:数据传递

模拟A协程和B协程,双方进行数据传输,在go语言中,协程数据的传输推荐的是使用通道,这里出现了一个新的数据类型:chan,使用它只需要关心的是:传输什么类型的数据,所以它的定义方法是:
var chan1 int = make(chan int)或者是简易法:chan1 := make(chan int)

通道的传递和接收语法

// 定义了一个传输数值型的通道
chan1 := make(chan int)

// 从通道中接收内容
x := <-chan1

// 往通道中传入一个值:1
chan1 <- 1

阻塞式通道

chan1 := make(chan int)

这种没有指定缓冲区的通道,默认是阻塞式通道。

理解阻塞:通道的使用有发送方和接收方两个角色。

  • 发送方发送数据时,当接收方没有准备好接收数据时,通道对于发送方是阻塞的。
  • 接收方,当通道中数据为空的时候,通道对于接收方是阻塞的。

看见一个非常形象的例子:京东快递送货上门,当没有暂存站点时,它会打电话给你,如果你方便接收,那它会给你送过来,如果你不能接收,则不会给你送过来。

常见错误,阻塞式通道兼任两个角色:

阻塞式非常注重发送方和接收方这两个角色,同个协程 不允许兼任两个身份
func main() {
   chan1 := make(chan int)
   // 接收
   x := <-chan1
   //发送
   chan1 <- 1
   fmt.Println(x)
}

改正,接收方改为新的协程;Ps:这里涉及到资源抢夺的问题,因为并没有使用WaitGroup

func main() {
   chan1 := make(chan int)
   go func() {
      // 接收
      x := <-chan1
      fmt.Println(x)
   }()
   //发送
   chan1 <- 1
}

常见错误,接收未准备好就发送数据:

先有接收状态才能往通道发送内容!这是阻塞式通道的原则。
func main() {
   chan1 := make(chan int)

   //发送
   chan1 <- 1

   go func() {
      // 接收
      x := <-chan1
      fmt.Println(x)
   }()
}

改正:把go协程放在发送注释上方即可。

常见错误,往里存了2个数

// 接收一个

chan1 := make(chan int)

chan1 <- 1
chan1 <- 2

新的输入无法在通道非空的情况下传入,即传送数据时,通道必须是空的。

非阻塞式通道

chan1 := make(chan int,1)

相比阻塞式通道,这里就多指明了一个参数,这个参数是一个数值型,表示的是缓存区,数据在通道中传输时可以保存1个,拿上面常见错误的最后一个例子来说:

// 接收一个

chan1 := make(chan int,1)

chan1 <- 1
chan1 <- 2

这里我们缓冲了一个,被接收了一个,所以再放值的时候不会报错。

非阻塞通道还有这些特点:

  • 只要缓存区运行,没有接收方也不会报错。
  • 因为缓存区的存在,打破了接收方和发送方的限制,在一个协程中发送和接收是被允许的,甚至不讲顺序。

这两点可以自己去尝试一下。

遍历通道的内容

方法一

使用无穷循环,配上判断;无穷循环不断的去执行获取chan中的内容,它会返回两个值,第一个值是实际值,第二个值是通道是否关闭,若通道没关闭则进行相应的判断

for {
   v, ok := <-chan1
   fmt.Println(v)
   fmt.Println(ok)
}

方法二

不用关心通道是否关闭,通道关闭时自动退出循环。

for v := range chan1 { // 优雅的从通道循环取值
   fmt.Println(v)
}

【重点】 方法一和方法二的取舍:

  • 方法一 不需要告知通道关闭,因为通道在垃圾回收的机制内。
  • 方法二 需要告诉通道关闭来控制退出循环

综合例子一:传输数字,直到...

往通道里传输数字,直到传到10位置
package main

import (
   "fmt"
   "sync"
)

var wg sync.WaitGroup

func main() {
   var chan1 chan int = make(chan int, 10)
   i := 0

   wg.Add(1)
   go func() {
      for v := range chan1 { //如果通道关闭会自动退出循环,如果没有告知通道关闭,会报错
         fmt.Println(v)
      }
      defer wg.Done()
   }()

   for {
      if i == 10 {
         // 告知通道关闭了
         close(chan1)
         break
      } else {
         chan1 <- i
         i++
      }
   }

   wg.Wait()
}

综合例子二:筛选偶数

package main

import (
   "fmt"
   "sync"
)

var wg sync.WaitGroup

func main() {
   var in chan []int = make(chan []int, 10)
   var out chan []int = make(chan []int, 10)
   x := make([]int, 0)
   x = append(x, 1, 2, 3, 4, 5, 6, 7)
   out <- x
   close(out) // 关闭通道

   wg.Add(1)
   go odd(in, out)

   for v := range in { //同样需要告诉通道关闭了
      fmt.Println(v)
   }
   //time.Sleep(time.Second * 2000)
   wg.Wait()
}

func odd(in chan<- []int, out <-chan []int) {
   tmp := make([]int, 0)
   for res := range out { // 用range需要告诉通道关闭了
      //fmt.Println(res)
      for _, value := range res {
         if value%2 == 0 {
            //fmt.Println(value)
            tmp = append(tmp, value)
         }
      }
   }
   in <- tmp
   close(in)
   wg.Done()
}

// 助记
//1. chan<- int是一个只能发送的通道,可以发送但是不能接收;
//2. <-chan int是一个只能接收的通道,可以接收但是不能发送。
目录
相关文章
|
9天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
45 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
29天前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
40 7
|
29天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
29天前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
101 71
|
28天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
105 67
|
1月前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
101 62
|
1月前
|
并行计算 安全 Go
Go语言中的并发编程:掌握goroutines和channels####
本文深入探讨了Go语言中并发编程的核心概念——goroutine和channel。不同于传统的线程模型,Go通过轻量级的goroutine和通信机制channel,实现了高效的并发处理。我们将从基础概念开始,逐步深入到实际应用案例,揭示如何在Go语言中优雅地实现并发控制和数据同步。 ####
|
4天前
|
算法 安全 Go
Go 语言中实现 RSA 加解密、签名验证算法
随着互联网的发展,安全需求日益增长。非对称加密算法RSA成为密码学中的重要代表。本文介绍如何使用Go语言和[forgoer/openssl](https://github.com/forgoer/openssl)库简化RSA加解密操作,包括秘钥生成、加解密及签名验证。该库还支持AES、DES等常用算法,安装简便,代码示例清晰易懂。
31 12
|
29天前
|
存储 Go
go语言中映射
go语言中映射
36 11
|
1月前
|
Go
go语言for遍历映射(map)
go语言for遍历映射(map)
35 12