返回顶部
首页 > 资讯 > 后端开发 > GO >Go语言之Goroutine与信道异常处理
  • 240
分享到

Go语言之Goroutine与信道异常处理

2024-04-02 19:04:59 240人浏览 薄情痞子
摘要

目录一、Goroutine1、启动一个 Goroutine2、Go 语言的GMP模型二、信道1、死锁2、单向信道3、for 循环信道4、缓冲信道5、WaitGroup6、Selec&

一、Goroutine

Go 协程可以看做成一个轻量级的线程Go 协程相比于线程的优势:

Goroutine 的成本更低大小只有 2 kb 左右,线程有几个兆。

Goroutine 会复用线程,比如说:我有 100 个协程,但是都是共用的的 3 个线程。

Goroutine 之间通信是通过 channel 通信的。(Go 推崇的是信道通信,而不推崇用共享变量通信)

1、启动一个 Goroutine


func test() {
   fmt.Println("go go go")
}

func main() {
   fmt.Println("主线程开始")
    
   // go 关键字开启 Goroutine,一个 Goroutine只占2kb左右
   go test() // 一个 go 就是一个协程,多个就是多个协程,也可以for循环起多个协程
   go test()
    
   time.Sleep(1*time.Second)  // Go 语言中主线程不会等待Goroutine执行完成,要等待它结束需要自己处理
   fmt.Println("主线程结束")
}

// 输出:
主线程开始
go go go
go go go
主线程结束

2、Go 语言的GMP模型

G:就是我们开起的 Goroutine

  • 它们会被相对平均的放到 P 的队列中

M:M 可以当成操作系统真正的线程,但是实际上是用户态的线程(用户线程)

  • 虽然 M 执行 G,但是实际上,M 只是映射到操作系统的线程上执行的
  • 然后操作系统的调度器把真正的操作系统的线程调度到CPU上执行

P:Processor 1.5版本以后默认情况是 CPU 核数(可以当做CPU核数)

  • P 会和 M 做交互,真正执行的是 M 在执行 G ,只不过 P 是在做调度
  • 一旦某个 G 阻塞了,那么 P 就会把它调度成下一个 G,放到 M 里面去执行

用户线程操作系统线程:

python 中,用户线程跟操作系统线程是 1:1 的对应关系

Go 语言中,用户线程和操作系统线程是 m:n 的关系

二、信道

信道(Channel)也就是 Go 协程之间的通信管道,一端发送一端接收。


func main() {
   // 1、定义 channel
   var c chan int

   // 2、管道的零值
   //———>空值为 nil 说明是引用类型,当做参数传递时,不需要取地址,改的就是原来的,需要初始化在使用
   fmt.Println(c)     // 输出:<nil>

   // 3、管道初始化
   c = make(chan int)
   go test(c)        // c 是引用类型直接传

   // 4、从信道中取值,信道默认不管放值还是取值,都是阻塞的
   count := <-c   // 阻塞
   fmt.Println(count)
   
   
}

func test(c chan int) {
   fmt.Println("GO GO GO")
   time.Sleep(1 * time.Second)
   // 5、往信道中放一个值,信道默认不管放值还是取值,都是阻塞的
   c <- 1 // 阻塞
}

// 输出:
<nil>
GO GO GO
1

1、死锁

Goroutine 给一个信道放值的时候,按理会有其他 Goroutine 来接收数据,没有的话就会形成死锁。


func main() {  
    c := make(chan int)
    c <- 1
}

// 报错:应为没有其他 Goroutine 从 c 中取值

2、单向信道

显而易见就是只能读或者只能写的信道

方式一:


func WriteOnly(c chan<- int) {
   c <- 1
}

func main() {
   write := make(chan<- int)  // 只写信道
   go WriteOnly(write)
   fmt.Println(<-write)      // 报错   ——>只写信道往外取就报错
}

方式二:


func WriteOnly(c chan<- int) {
   c <- 1
   // <-c       // 报错
}

func main() {
   write := make(chan int)    // 定义一个可读可写信道
   go WriteOnly(write)        // 传到函数中就成了只写信道,在Goroutine中只负责写,不能往外读
   fmt.Println(<-write)     // 主协程读
}

3、for 循环信道

for 循环循环信道,如果不关闭,会报死锁,如果关闭了,放不进去,循环结束。


func producer(chnl chan int) {
   for i := 0; i < 10; i++ {
      chnl <- i   // i 放入信道
   }
   close(chnl)      // 关闭信道
}

func main() {
   ch := make(chan int)
   go producer(ch)
   // 循环获取信道内容
   for value := range ch {
      fmt.Println(value)
   }
}



4、缓冲信道

在默认情况下信道是阻塞的,缓冲信道也就是说我信道里面可以缓冲一些东西,可以不阻塞了。

只有在缓冲已满的情况,才会阻塞信道

只有在缓冲为空的时候,才会阻塞主缓冲信道接收数据


func main() {
   // 指定的数字就是缓冲大小
   var c chan int = make(chan int, 3) // 无缓冲信道数字是0
   c <- 1
   c <- 2
   c <- 3
   c <- 4     // 缓冲满了,死锁

   <-c
   <-c
   <-c
   <-c          // 取空了,死锁

   fmt.Println(len(c))       // 长度:目前放了多少
   fmt.Println(cap(c))       // 容量:可以最多放多少
}

5、WaitGroup

等待所有 Goroutine 执行完成


func process1(i int, wg *sync.WaitGroup) {
   fmt.Println("started Goroutine ", i)
   time.Sleep(2 * time.Second)
   fmt.Printf("Goroutine %d ended\n", i)
   // 3、一旦有一个完成,减一
   wg.Done()
}

func main() {
   var wg sync.WaitGroup    // 没有初始化,值类型,当做参数传递,需要取地址

   for i := 0; i < 10; i++ {
      wg.Add(1)      // 1、启动一个 Goroutine,add 加 1
      go process1(i, &wg)     // 2、把wg传过去,因为要改它并且它是值类型需要取地址传过去
   }

   wg.Wait()  // 4、一直阻塞在这,直到调用了10个 Done,计数器减到零
}

6、Select

Select 语句用于在多个发送 / 接收信道操作中进行选择。

例如:我要去爬百度,我发送了三个请求去,可能有一些网络原因,或者其他原因,不一定谁先回来,Select 选择就是谁先回来我先用谁。

场景一:对性能极致的要求,我就可以选择一个最快的线路执行我最快的功能,就可以用Select来做

场景二:我去拿这个数据的时候,不是一直等在这里,而是我可以干一点别的事情,使用死循环 Select 的时候加上 default 去做其他事情。


// 模拟去服务器去取值
func server(ch chan string) {
   time.Sleep(3 * time.Second)
   ch <- "from server"
}

func main() {
   output1 := make(chan string)
   output2 := make(chan string)
    
   // 开起两个协程执行 server
   go server(output1)
   go server(output2)

   select {
   case s1 := <-output1:  // 阻塞,谁先回来就执行谁
      fmt.Println(s1, "output1")
   case s2 := <-output2:  // 阻塞,谁先回来就执行谁
      fmt.Println(s2, "output2")
   }
}

7、Mutex

使用锁的场景:多个 Goroutine 通过共享内存来实现数据通信,就会出现并发安全的问题,并发安全的问题就需要加锁。

临界区:当程序并发运行时修改共享资源的代码,也就同一块内存的变量的时候,这些修改的资源的代码就称为临界区。

如果在任意时刻只允许一个 Goroutine 访问临界区,那么就可以避免竞争条件,而使用 Mutex(锁) 可以实现

不加用锁的情况下:


var x = 0   //全局,各个 Goroutine 都可以拿到并且操作

func increment(wg *sync.WaitGroup) {
   x = x + 1
   wg.Done()
}

func main() {
   var w sync.WaitGroup
   for i := 0; i < 1000; i++ {
      w.Add(1)
      go increment(&w)
   }
   w.Wait()

   fmt.Println("最终的值:", x)
}

// 输出:理想情况下是1000,因为并发有安全的问题,所以数据乱了
最终的值: 978

加锁的情况:


var x = 0   //全局,各个 Goroutine 都可以拿到并且操作

func increment(wg *sync.WaitGroup, m *sync.Mutex) {
   m.Lock()    // 加锁
   x = x + 1 // 同一时间只能有一个 Goroutine 执行
   m.Unlock()  // 解锁
   wg.Done()
}
func main() {
   var w sync.WaitGroup
   var m sync.Mutex    // 因为是个值类型,函数传递需要传地址
   fmt.Println(m)     // 输出:{0 0} ——>值类型

   for i := 0; i < 1000; i++ {
      w.Add(1)
      go increment(&w, &m)
   }
   w.Wait()
   fmt.Println("最终的值:", x)
}

// 输出:
最终的值: 1000

使用信道来实现:


var x = 0

func increment(wg *sync.WaitGroup, ch chan bool) {
   ch <- true   // 缓冲信道放满了,就会阻塞。
   x = x + 1
   <-ch         // 执行完了就取出
   wg.Done()
}
func main() {
   var w sync.WaitGroup
   ch := make(chan bool, 1)   // 定义了一个有缓存大小为1的信道
   for i := 0; i < 1000; i++ {
      w.Add(1)
      go increment(&w, ch)
   }

   w.Wait()
   fmt.Println("最终的值:", x)
}

// 输出:
最终的值:1000

总结:不同 Goroutine 之间传递数据的方式:共享变量、信道。

如果是修改变量,倾向于用 Mutex

如果是 Goroutine 之间通信,倾向于用信道

三、异常处理

defer:延时执行,并且即便程序出现严重错误,也会执行


func main() {
 defer fmt.Println("我最后执行")
 defer fmt.Println("我倒数第三执行")
 fmt.Println("我先执行")
    
 var a []int
 fmt.Println(a[10])   // 报错

 fmt.Println("后执行") // 不会执行了
}


// 输出:
我先执行
我倒数第三执行
我最后执行
panic: runtime error: index out of range [10] with length 0
panic:主动抛出异常

recover:恢复程序,继续执行

func f1() {
   fmt.Println("f1 f1")
}


func f2() {
   defer func() {        // 这个匿名函数永远会执行

      //如果没有错误,执行 recover 会返回 nil,如果有错误执行 recover 会返回错误信息
      if error := recover(); error != nil {
         // 表示出错了,打印一下错误信息,程序恢复了,继续执行
         fmt.Println(error)
      }
      fmt.Println("我永远会执行,不管是否出错")
   }()
    
   fmt.Println("f2 f2")
   panic("主动抛出错误")
}


func f3() {
   fmt.Println("f3 f3")
}


func main() {
   //捕获异常,处理异常,让程序继续运行
   f1()
   f2()
   f3()
}

Go 语言异常捕获与 Python 异常捕获对比

Python:


try:
 可能会错误的代码
except Exception as e:
 print(e)
finally:
 无论是否出错,都会执行的代码


Go :


defer func() {
 if error:=recover();error!=nil{
  // 错误信息 error
  fmt.Println(error)
 }
    
 相当于finally,无论是否出错,都会执行的代码
    
}()

可能会错误的代码

到此这篇关于Go语言之Goroutine与信道异常处理的文章就介绍到这了,更多相关Go语言Goroutine与信道异常处理内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

您可能感兴趣的文档:

--结束END--

本文标题: Go语言之Goroutine与信道异常处理

本文链接: https://lsjlt.com/news/154404.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Go语言之Goroutine与信道异常处理
    目录一、Goroutine1、启动一个 Goroutine2、Go 语言的GMP模型二、信道1、死锁2、单向信道3、for 循环信道4、缓冲信道5、WaitGroup6、Selec&...
    99+
    2024-04-02
  • go语言编程之select信道处理示例详解
    目录select信道处理fibonacci数列监听select监听协程select信道处理 注意:有default就不会阻塞 package main func main() { ...
    99+
    2024-04-02
  • Go语言错误处理异常捕获+异常抛出
    目录一、error变量可以做什么1.定义一个error变量2.错误的处理3.做函数返回值4.做函数参数二、模拟异常的捕获与抛出1.defer简介2.使用recover模拟异常的捕获3...
    99+
    2024-04-02
  • Go语言异常处理案例解析
    异常处理 程序运行时,发生的不被期望的事件,它阻止了程序按照程序员的预期正常执行,这就是异常 golang中提供了两种处理异常的方式 一种...
    99+
    2024-04-02
  • Go语言异常处理的示例分析
    这篇文章主要为大家展示了“Go语言异常处理的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Go语言异常处理的示例分析”这篇文章吧。异常处理程序运行时,发生的不被期望的事件,它阻止了程序按...
    99+
    2023-06-20
  • 深入探讨:Go语言中goroutine的并发编程之道
    在当前软件开发领域中,对于并发编程的需求越来越迫切。随着硬件技术的发展,多核处理器已经成为主流,而利用并发编程可以充分发挥多核处理器的潜力,提高系统的性能和响应速度。Go语言作为一门并...
    99+
    2024-03-13
    go语言 并发
  • 如何处理Go语言中的异常处理问题?
    如何处理Go语言中的异常处理问题?Go语言是一种由谷歌开发的编程语言,它以其简洁、高效、并发安全的特点受到了广泛的关注和应用。在日常的开发工作中,异常处理是一个不可避免的问题。本文将介绍一些在Go语言中处理异常的方法,并提供具体的代码示例。...
    99+
    2023-10-22
    Go语言 异常处理 问题处理
  • 探讨:go语言与golang的差异与共同之处
    深入解析:Go语言和Golang的异同,需要具体代码示例 Go语言是一种相对较新的编程语言,于2007年由Google开发推出,是一种静态类型、编译型的高级编程语言。随着Go语言的快速发展,人们开始普遍使用"...
    99+
    2024-01-20
    Golang Go语言 异同
  • Go语言学习教程之goroutine和通道的示例详解
    目录goroutine通道Range 和 CloseSelect官方留的两道练习题等价的二叉树网络爬虫源码地址goroutine goroutine是由Go运行时管理的轻量级线程。 ...
    99+
    2024-04-02
  • 使用Go语言进行高级异常处理
    在 go 中,处理高级异常的最佳实践包括:使用 try-catch-finally 语句处理异常。创建自定义错误类型以提供更有意义的异常信息。传递错误值以跟踪异常来源。使用恐慌和恢复来处...
    99+
    2024-04-15
    go 异常处理 go语言
  • 不同语言中错误处理与Go语言错误处理的差异
    不同编程语言的错误处理方式不同,如 javascript 使用 try...catch 块,java 使用异常类。go 语言使用错误值,一种表示错误详细信息的接口类型。函数可以通过返回错...
    99+
    2024-04-12
    go 错误处理 python go语言
  • 你知道GO语言与UNIX索引之间的自然语言处理的联系吗?
    当今的计算机科学领域中,自然语言处理(NLP)一直是一个备受瞩目的领域。它有助于让计算机理解人类语言,并为我们提供更加智能化的体验。在NLP的基础上,我们可以构建各种各样的应用,例如语音识别、机器翻译、情感分析等等。GO语言和UNIX索引则...
    99+
    2023-08-14
    unix 索引 自然语言处理
  • Go语言异常处理(Panic和recovering)用法详解
    目录基本语法PanicRecover示例一:recover()使用方法实例二:panic()使用方法基本语法 异常处理是程序健壮性的关键,往往开发人员的开发经验的多少从异常部分处理上...
    99+
    2024-04-02
  • C++ 函数异常处理如何与其他语言的异常处理机制相比较?
    c++++ 函数异常处理采用函数 try-catch 块,抛出的异常立即传播到调用函数中,可通过 catch 块捕获和处理。java 和 python 的异常处理分别使用 try-cat...
    99+
    2024-04-15
    c++ 异常处理 python
  • Go语言与C语言有哪些不同之处
    这篇文章将为大家详细讲解有关Go语言与C语言有哪些不同之处,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。C 的语法首先,先看看 C 的语法。C 采用了一种聪明而不同寻常的声明语法。声明变量时,只需写出一个...
    99+
    2023-06-16
  • Python之异常处理、模块与包
    MarkdownPad Document程序中错误分成两种语法错误:过不了Python解释器逻辑错误异常处理什么是异常处理Python解释器检测到错误,触发异常,在发生异常时捕捉异常,如果捕捉成功则进入另外一个处理分支,是程序不会崩溃,这就...
    99+
    2023-01-31
    模块 异常 Python
  • plsql的错误信息与异常处理
    7 错误消息与异常处理7.1 异常的介绍(1) 处理异常分为三个步骤:  A 声明异常  B 引发异常  C 处理异常(2) 异常的特征  A 错误类型: ORA-x...
    99+
    2024-04-02
  • GO语言中err接口及defer延迟异常处理分析
    目录err接口panic函数defer延迟defer与匿名函数结合使用recover防止程序中断err接口 Go语言引入了一个关于错误处理的标准模式,即error接口,它是Go语言内...
    99+
    2024-04-02
  • GO语言中err接口及defer延迟异常怎么处理
    这篇“GO语言中err接口及defer延迟异常怎么处理”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“GO语言中err接口及d...
    99+
    2023-06-30
  • Go语言与其他语言的区别和相似之处
    Go语言与其他语言的区别和相似之处 Go语言作为一种静态类型的编程语言,自诞生以来就备受程序员们的青睐。它的设计初衷是简洁、高效、并发性强,因此与其他语言相比有着许多独特之处,同时也有...
    99+
    2024-03-07
    优化 并发 区别:类型 相似:语法 go语言
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作