Go中阻塞以及非阻塞操作实现(Goroutine和main Goroutine)

 更新时间:2024年05月15日 10:28:47   作者:vSeanere  
本文主要介绍了Go中阻塞以及非阻塞操作实现(Goroutine和main Goroutine),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

main函数本身也是一个Goroutine。

在Go语言中,main函数是程序的入口点。当程序启动时,Go运行时会创建一个名为main的Goroutine,并在其中执行main函数的代码。这个main Goroutine 是程序中的主要执行线程,它负责执行main函数中的指令和操作。

与其他用户创建的Goroutine类似,main Goroutine 也是并发执行的。它可以与其他Goroutine同时执行,利用多核处理器的并行性。

main Goroutine 的生命周期与程序的生命周期相同。当main函数执行完成或显式调用os.Exit函数终止程序时,main Goroutine 会随之结束。其他正在执行的Goroutine也会被终止,程序会退出。

因此,我们可以把main函数看作是程序中的一个特殊的Goroutine,它是程序的入口和控制中心。通过在main函数内部启动其他Goroutine,我们可以实现多个任务的并发执行,充分利用Go语言的并发能力。

Go语言中,阻塞和非阻塞是用来描述对某个操作的等待行为的。

阻塞(Blocking):当执行一个操作时,如果该操作无法立即完成,程序会被阻塞,即暂停执行,直到该操作完成或满足某个条件。在阻塞状态下,程序无法继续执行其他任务,直到阻塞的操作完成。例如,当我们读取一个通道(channel)时,如果通道中没有可读取的数据,读取操作将会阻塞,直到有数据可读。

package main

import (
	"fmt"
	"time"
)

func main() {
	ch := make(chan int) // 创建一个通道

	go func() {
		time.Sleep(2 * time.Second)
		ch <- 42 // 向通道发送数据
	}()

	fmt.Println("Waiting for data...")
	data := <-ch // 从通道接收数据,如果通道中没有数据,该行会阻塞等待
	fmt.Println("Received data:", data)
}

在上述示例中,我们创建了一个通道ch,并在一个Goroutine中等待2秒后向通道发送了数据(你可以尝试修改time.Sleep函数后的数字,分别运行来看效果)。在主Goroutine中,我们尝试从通道ch中接收数据。由于通道中没有数据可用,接收操作会被阻塞,直到数据发送到通道为止。

非阻塞(Non-blocking):当执行一个操作时,如果该操作无法立即完成,程序不会被阻塞,而是立即返回,不管操作是否成功或者完成。通过使用非阻塞操作,程序可以继续执行其他任务,而不需要等待阻塞的操作完成。例如,使用非阻塞的方式读取通道时,如果通道中没有数据可读,读取操作会立即返回一个错误或默认值。

package main

import (
	"fmt"
	"time"
)

func main() {
	ch := make(chan int) // 创建一个通道

	go func() {
		time.Sleep(2 * time.Second)
		ch <- 42 // 向通道发送数据
	}()

	fmt.Println("Waiting for data...")
	select {
	case data := <-ch: // 尝试从通道接收数据
		fmt.Println("Received data:", data)
	default:
		fmt.Println("No data available")
	}
}

在上述示例中,我们使用select语句来尝试从通道ch中接收数据。如果通道中没有可用的数据,select语句会立即执行default分支,输出"No data available",而不会阻塞等待。这样,程序可以继续执行其他任务,而不需要等待数据的到达。

阻塞和非阻塞是用来描述对于某个操作的等待行为的概念。阻塞表示当一个操作无法立即完成时,程序会暂停执行,直到操作完成。非阻塞表示当一个操作无法立即完成时,程序会立即返回,继续执行其他任务。在Go语言中,通常使用通道操作来展示阻塞和非阻塞的概念。

到此这篇关于Go中阻塞以及非阻塞操作实现(Goroutine和main Goroutine)的文章就介绍到这了,更多相关Go 阻塞 非阻塞内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Go语言实现多协程文件下载器的过程详解

    Go语言实现多协程文件下载器的过程详解

    这篇文章主要介绍了Go语言实现多协程文件下载器的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2024-08-08
  • Golang Socket Server自定义协议的简单实现方案

    Golang Socket Server自定义协议的简单实现方案

    这篇文章主要介绍了Golang Socket Server自定义协议的简单实现方案,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • golang给函数参数设置默认值的几种方式小结(函数参数默认值

    golang给函数参数设置默认值的几种方式小结(函数参数默认值

    在日常开发中我们有时候需要使用默认设置,下面这篇文章主要给大家介绍了关于golang给函数参数设置默认值的几种方式小结的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-01-01
  • Golang实践指南之获取目录文件列表

    Golang实践指南之获取目录文件列表

    在搭建项目中一般都会有确定项目根目录的绝对路径的需求,下面这篇文章主要给大家介绍了关于Golang实践指南之获取目录文件列表的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-01-01
  • Go语言操作Excel利器之excelize类库详解

    Go语言操作Excel利器之excelize类库详解

    Excelize是Go语言编写的用于操作Office Excel文档基础库,基于ECMA-376,ISO/IEC 29500国际标准,可以使用它来读取、写入由Excel 2007及以上版本创建的电子表格文档,下面这篇文章主要给大家介绍了关于Go语言操作Excel利器之excelize类库的相关资料,需要的朋友可以参考下
    2022-10-10
  • Go常用标准库之fmt的简介与使用详解

    Go常用标准库之fmt的简介与使用详解

    fmt 是 Go 语言中的一个常用标准库,它用于格式化输入和输出数据,这篇文章主要为大家介绍了fmt的基本使用,感兴趣的小伙伴可以跟随小编一起学习一下
    2023-10-10
  • 基于Golang container/list实现LRU缓存

    基于Golang container/list实现LRU缓存

    Least Recently Used (LRU) ,即逐出最早使用的缓存,这篇文章主要为大家介绍了如何基于Golang container/list实现LRU缓存,感兴趣的可以了解下
    2023-08-08
  • Golang基础学习之map的示例详解

    Golang基础学习之map的示例详解

    哈希表是常见的数据结构,有的语言会将哈希称作字典或者映射,在Go中,哈希就是常见的数据类型map,本文就来聊聊Golang中map的相关知识吧
    2023-03-03
  • 一文掌握go的sync.RWMutex锁

    一文掌握go的sync.RWMutex锁

    这篇文章主要介绍了一文掌握go的sync.RWMutex锁,本文是为了在面试中能快速口述RW锁,并非为了完整解答RW锁的机制,需要的朋友可以参考下
    2023-03-03
  • 一文详解Go语言fmt标准库的常用占位符使用

    一文详解Go语言fmt标准库的常用占位符使用

    这篇文章主要为大家详细介绍了Go语言中fmt标准库的常用占位符及其简单使用,文中的示例代码讲解详细,对我们学习Go语言有一定的帮助,需要的可以参考一下
    2022-12-12

最新评论