Golang并发编程-协程goroutine初体验

2024-05-25 01:12

本文主要是介绍Golang并发编程-协程goroutine初体验,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 一、Goroutine适合的使用场景
  • 二、Goroutine的使用
    • 1. 协程初体验
  • 三、WaitGroup
    • WaitGroup 案例一
    • WaitGroup 案例二
  • 总结


前言

学习Golang一段时间了,一直没有使用过goroutine来提高程序执行效率,在一些特殊场景下,还是有必须开启协程提升体验的,打算整理几篇关于协程的原理的文章和案例,结合工作场景将协程使用起来。


一、Goroutine适合的使用场景

并发执行任务: Goroutine 可用于同时执行多个任务,提高程序的性能。
非阻塞 I/O 操作: 在进行 I/O 操作时,可以使用 goroutine 确保其他任务继续执行,而不是同步等待 I/O 完成。
事件驱动编程: Goroutine 可用于处理事件,如监听 HTTP 请求、处理用户输入等。
并发算法: 实现一些需要并行计算的算法,通过 goroutine 可以更轻松地管理并发执行的部分。
定时任务: 使用 goroutine 和定时器可以实现定时执行的任务。

二、Goroutine的使用

1. 协程初体验

一个 Go 程序的入口通常是 main 函数,程序启动后,main 函数最先运行,我们称之为 main goroutine。

在 main 中或者其下调用的代码中才可以使用 go + func() 的方法来启动协程。

main 的地位相当于主线程,当 main 函数执行完成后,这个线程也就终结了,其下的运行着的所有协程也不管代码是不是还在跑,也得乖乖退出。

package mainimport "fmt"func mytest() {fmt.Println("hello, go")
}func main() {// 启动一个协程go mytest()fmt.Println("hello, world")
}

因此上面这段代码运行完,只会输出 hello, world ,而不会输出hello, go(因为协程的创建需要时间,当 hello, world打印后,协程还没来得及并执行)

当我在代码中加入一行 time.Sleep 输出就符合预期了。

package mainimport ("fmt""time"
)func mytest() {fmt.Println("hello, go")
}func main() {// 启动一个协程go mytest()fmt.Println("hello, world")time.Sleep(time.Second)
}

输出结果对比如下:

[root@work day01]# go run main.go 
hello, world
[root@work day01]# go run main.go 
hello, world
[root@work day01]# go run main.go 
hello, world
hello, go
[root@work day01]# 

三、WaitGroup

在上面的例子部分,为了保证 main goroutine 在所有的 goroutine 都执行完毕后再退出,我使用了 time.Sleep 这种简单的方式。这种方式在demo程序是可以接受的。但是当实际开发过程中,不同场景下,Sleep 多少时间呢,是无法预测的。
因此,Sleep这种方式还是尽量不要用了,下面介绍下sync包提供的WaitGroup类型。

WaitGroup 案例一

代码如下(示例):

package mainimport ("fmt""sync"
)func printNumbers(wg *sync.WaitGroup) {defer wg.Done() // 当某个子协程完成后,可调用此方法,会从计数器上减一,通常可以使用 defer 来调用。for i := 1; i <= 5; i++ {fmt.Printf("%d ", i)}
}func printLetters(wg *sync.WaitGroup) {defer wg.Done() // 当某个子协程完成后,可调用此方法,会从计数器上减一,通常可以使用 defer 来调用。for char := 'a'; char <= 'e'; char++ {fmt.Printf("%c ", char)}
}func main() {var wg sync.WaitGroup   wg.Add(2) // 初始值为0,你传入的值会往计数器上加,这里直接传入你子协程的数量go printNumbers(&wg)go printLetters(&wg)wg.Wait() // 阻塞当前协程,直到实例里的计数器归零。
}

结果如下:

[root@work day01]# go run main2.go 
a b c d e 1 2 3 4 5

WaitGroup 案例二

package mainimport ("fmt""io/ioutil""net/http""sync"
)func fetch(url string, wg *sync.WaitGroup) {defer wg.Done()response, err := http.Get(url)if err != nil {fmt.Printf("Error fetching %s: %v\n", url, err)return}defer response.Body.Close()body, err := ioutil.ReadAll(response.Body)if err != nil {fmt.Printf("Error reading response body from %s: %v\n", url, err)return}fmt.Printf("Length of %s: %d\n", url, len(body))
}func main() {var wg sync.WaitGroupurls := []string{"https://www.baidu.com", "https://cloud.tencent.com/", "https://www.qq.com/"}for _, url := range urls {wg.Add(1)go fetch(url, &wg)}wg.Wait()
}

输出结果如下:

[root@work day01]# go run main3.go 
Length of https://www.qq.com/: 328
Length of https://www.baidu.com: 2443
Length of https://cloud.tencent.com/: 235026

总结

本节内容,介绍了Goroutine的使用,为了保证 main goroutine 在所有的 goroutine 都执行完毕后再退出,我们又学习了WaitGroup。目前呢,因为我们没有任何的数据交换,仅仅是开启协程执行并发的任务,因此没有用到信道。后面遇到复杂一些并发场景,我们的goroutine通信就要用到信道的概念。这里我们下一节介绍。

这篇关于Golang并发编程-协程goroutine初体验的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1000059

相关文章

Web服务器-Nginx-高并发问题

《Web服务器-Nginx-高并发问题》Nginx通过事件驱动、I/O多路复用和异步非阻塞技术高效处理高并发,结合动静分离和限流策略,提升性能与稳定性... 目录前言一、架构1. 原生多进程架构2. 事件驱动模型3. IO多路复用4. 异步非阻塞 I/O5. Nginx高并发配置实战二、动静分离1. 职责2

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

Spring Security 前后端分离场景下的会话并发管理

《SpringSecurity前后端分离场景下的会话并发管理》本文介绍了在前后端分离架构下实现SpringSecurity会话并发管理的问题,传统Web开发中只需简单配置sessionManage... 目录背景分析传统 web 开发中的 sessionManagement 入口ConcurrentSess

C#异步编程ConfigureAwait的使用小结

《C#异步编程ConfigureAwait的使用小结》本文介绍了异步编程在GUI和服务器端应用的优势,详细的介绍了async和await的关键作用,通过实例解析了在UI线程正确使用await.Conf... 异步编程是并发的一种形式,它有两大好处:对于面向终端用户的GUI程序,提高了响应能力对于服务器端应

MySQL中处理数据的并发一致性的实现示例

《MySQL中处理数据的并发一致性的实现示例》在MySQL中处理数据的并发一致性是确保多个用户或应用程序同时访问和修改数据库时,不会导致数据冲突、数据丢失或数据不一致,MySQL通过事务和锁机制来管理... 目录一、事务(Transactions)1. 事务控制语句二、锁(Locks)1. 锁类型2. 锁粒

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer

C# async await 异步编程实现机制详解

《C#asyncawait异步编程实现机制详解》async/await是C#5.0引入的语法糖,它基于**状态机(StateMachine)**模式实现,将异步方法转换为编译器生成的状态机类,本... 目录一、async/await 异步编程实现机制1.1 核心概念1.2 编译器转换过程1.3 关键组件解析

全面解析Golang 中的 Gorilla CORS 中间件正确用法

《全面解析Golang中的GorillaCORS中间件正确用法》Golang中使用gorilla/mux路由器配合rs/cors中间件库可以优雅地解决这个问题,然而,很多人刚开始使用时会遇到配... 目录如何让 golang 中的 Gorilla CORS 中间件正确工作一、基础依赖二、错误用法(很多人一开