【Hello Go】Go语言并发编程

并发编程

    • 概述
      • 基本概念
      • go语言的并发优势
    • goroutine
      • goroutine是什么
      • 创建goroutine
      • 如果主goroutine退出
      • runtime包
        • Gosched
        • Goexit
        • GOMAXPROCS
    • channel
      • 无缓冲的channel
      • 有缓冲的channel
      • range和close
      • 单向channel
    • 定时器
      • Timer
      • Ticker
    • Select
      • 超时

概述

基本概念

并行和并发概念

  • 并行 :在同一时刻 有多条指令在多个编译器上
  • 并发 :在同一时刻 只能有一条指令执行 但是多个进程指令被快速的轮换执行 使得在宏观上有多个进程被同时执行的效果

如果我们把它具象化成现实中的概念

  • 并行就是同一时刻 两个队列使用两台咖啡机
  • 并发就是同一时刻 两个队列使用一台咖啡机

go语言的并发优势

第一 Go语言在语言层面上天然支持并发 (不像某个语言 23版本才勉强上线)

第二 并发编程的内存管理都是十分复杂的 而Go语言支持GC即 垃圾回收机制


Go语言为了支持并发编程而内置的上层API是基于 CSP (顺序通信进程) 模型 这就意味着显示锁都是可以避免的 而Go语言通过相册安全的通道发送和接受数据以实现同步 这大大简化了并发程序的编写

一般情况下 一个普通的桌面计算机系统跑十几二十个线程就会有点负载了 但是同样的这台计算机却能轻松的让成百上千甚至过万个 goroutine进行资源竞争

goroutine

goroutine是什么

goroutine是Go并发设计的核心 说到底 其实它是协程 但是它比线程更小 十几个goroutine在底层的体现可能是几个线程

Go语言内部帮你实现了帮你实现了这些goroutine之间的内存共享 执行它只需要极少的栈内存 大概(4~5kb) 正因为如此 可以同时运行成千上万个goroutine任务

goroutine比thread更高效 更简单 更轻便

创建goroutine

只需要在函数调用之前添加go关键字 就可以创建并发执行单元 开发人员无需了解任何细节 调度器会自动将其安排到合适的系统线程上执行

在并发编程里 我们通常想将一个过程切分成几块 并且然后让每个goroutine负责它的一部分 当一个程序运行时 它的主函数即在一个单独的goroutine中执行 我们把它叫做 main goroutine

而新的goroutine使用go语句来创建

代码演示如下

func testnewgor() {for i := 0; i < 5; i++ {fmt.Println("new goroutine say :", i)time.Sleep(time.Second)}
}func main() {go testnewgor()for i := 0; i < 5; i++ {fmt.Println("main goroutine say :", i)time.Sleep(time.Second)}
}

运行这段代码之后我们会发现主协程 新协程会同时打印语句

如果主goroutine退出

如果说主goroutine推出了 并不会有类似linux中孤儿进程的概念 其他的goroutine也会立即退出

runtime包

Gosched

runtime.gosched() 用于让出CPU时间片 让出当前协程的执行权限 调度器会安排其他等待的任务执行 并在下次的某个时刻从该位置开始恢复执行

这就像接力赛一样 A跑了一段时间遇到代码runtime.gosched() 之后将接力棒交给B 之后B跑了一段时间遇到代码runtime.gosched()之后将接力棒交给A

下面是示例代码

func main() {go func() {for i := 0; i < 5; i++ {runtime.Gosched()fmt.Println("world")}}()// main gorotinuefor i := 0; i < 5; i++ {fmt.Println("hello")runtime.Gosched()}// 最后结果为  hello  world  hello world ... ...}
Goexit

调用Goexit函数将会立即终止当前goroutine执行 调度器会确保所有的defer调用被执行

下面是示例代码演示

	go func() {defer fmt.Println("this is A")runtime.Goexit()defer fmt.Println("this is B")fmt.Println("this is C")}()  // 只会打印 this is A   因为后面的延时调用语句还没来得及执行协程就退出了 // 不让主协程退出  观察其他携程的掩饰效果for {}
GOMAXPROCS

GOMAXPROCS在Go语言中是一个环境变量 它表示可以Go语言可以并发的最大核心数

如果是 runtime.GOMAXPROCS(size int) 函数 我们有两种用法

  • 第一种是将参数设置0 此时会返回我们当前的最大核心数
  • 第二种是将参数设置为其他正整数 此时核心会变为我们设置的值

channel

它和map类似 channel也是一个对于make创建的底层数据结构的引用

当我们复制了一个channel用于函数传参时 我们只是拷贝了一个channel引用 因此调用者和被调用者将使用同一个channel对象 和其他的引用类型一样 channel的零值也是nil

定义一个channel时 我们也需要定义发送到chanel值的类型 channel可以使用内置的make()函数来实现

make(chan Type) //等价于make(chan Type, 0)
make(chan Type, capacity)

当capacity等于0的时候 是无缓冲阻塞式读写的

当capacity大于0的时候 是有缓冲非阻塞的 直到写入的数据大于capacity才会阻塞住

channel通过操作符<-来接收和发送数据 发送和接收数据语法如下

channel <- value // 发送value到channel 
<- channel // 接受并且丢弃所有数据 
x := <- channel  // 从channel接受数据 并且赋值给x
x , ok := <- channel  // 功能同上 不过增加了一个bool类型的数据来检查通道是否关闭或者是否为空 

在默认情况下 channel接受和发送数据都是阻塞的 除非另一端已经准备好了 这就让goroutine的同步变得简单 不需要显示的lock了

	c := make(chan int)go func() {fmt.Println("子协程正在运行")defer fmt.Println("子协程已结束")c <- 666}()fmt.Println("主协程正在运行")x := <-ctime.Sleep(time.Second)fmt.Println("子协程发送的值为", x)

无缓冲的channel

无缓冲的通道(unbuffered channel)是指在接收前没有能力保存任何值的通道

这种类型的通道要求发送 goroutine 和接收 goroutine 同时准备好,才能完成发送和接收操作。如果两个goroutine没有同时准备好,通道会导致先执行发送或接收操作的 goroutine 阻塞等待。

这种对通道进行发送和接收的交互行为本身就是同步的。其中任意一个操作都无法离开另一个操作单独存在。

我们上面的代码就是一个无缓冲channel 这里为了方便大家理解再发一遍

	c := make(chan int)go func() {fmt.Println("子协程正在运行")defer fmt.Println("子协程已结束")c <- 666}()fmt.Println("主协程正在运行")x := <-ctime.Sleep(time.Second)fmt.Println("子协程发送的值为", x)

有缓冲的channel

有缓冲的channel创建方式如下

make(chan Type, capacity)

此时它阻塞的方式也发生了变化

  • 如果缓冲区满了并且还在写数据此时会写入阻塞
  • 如果缓冲区空了并且还在读数据此时会读取阻塞

range和close

我们可以通过close来关闭一个channel

close (chan)
  • channel 不像文件一样需要经常去关闭 只有当你确实没有任何发送数据了 或者要结束range循环才关闭
  • 关闭之后无法再发送任何的数据 发数据会引发panic异常
  • 关闭后可以接受数据
  • 接受数据会阻塞住

此外我们还可以通过range迭代来获取数据 一旦管道关闭 range循环就会结束

单向channel

默认情况下,通道是双向的,也就是,既可以往里面发送数据也可以同里面接收数据

但是,我们经常见一个通道作为参数进行传递而值希望对方是单向使用的,要么只让它发送数据,要么只让它接收数据,这时候我们可以指定通道的方向

单向channel变量的声明非常简单,如下:

var ch1 chan int       // ch1是一个双向的管道
var ch2 chan<- float64 // ch2只能往里写入float64数据
var ch3 <-chan int     // ch3只能用于接受int类型的数据
  • chan<- 表示数据进入管道,要把数据写进管道,对于调用者就是输出。
  • <-chan 表示数据从管道出来,对于调用者就是得到管道的数据,当然就是输入

我们可以将channel隐式的转化为单向队列只收或者只发 不能将单向的channel转化为普通channel

转换的语法如下

c := make(chan int, 3)
var send chan<- int = c // send-only
var recv <-chan int = c // receive-only

下面是完整的使用代码

func recv(out <-chan int) {for x := range out {fmt.Println(x)}
}func send(in chan<- int) {for i := 0; i < 5; i++ {in <- i * 100}close(in)
}func main() {c := make(chan int, 3)go send(c)recv(c)time.Sleep(3 * time.Second)
}

定时器

Timer

timer是一个定时器 代表未来的一个单一事件 你可以告诉timer这个时间要等待的时间 它会提供一个channel 在将来的那个时间 channel提供了一个时间值

下面是示例代码

func main() {// 创建定时器 两秒后定时器就会像自己的c字节发送一个time.TIME类似的元素值timer1 := time.NewTimer(2 * time.Second)t1 := time.Now() // 当前时间fmt.Printf("t1 : %v\n", t1)t2 := <-timer1.Cfmt.Println("t2:", t2)
}

我们在创建定时器之后的两秒钟会收到一个时间 之后我们可以将该时间和现在的时间对比一下 我们发现正好相差了两秒

Ticker

Ticker是一个定时触发的计时器 它会以一个间隔往channel中发送一个事件 而channel的接收者可以以固定的时间间隔从channel中读取事件

下面是示例代码

func main() {// 创建一个定时器 每隔一秒像channel中发送一个事件ticker := time.NewTicker(time.Second * 1)i := 0go func() {for i = 0; i < 5; i++ {<-ticker.Cprintln("goroutine say : ", i)}// 最后关闭tickerticker.Stop()}()for {}
}

Select

Go语言提供了一个关键字select 通过select可以监听channel上的数据流动

select的用法和switch十分相似 由select选择一个新的模块 之后每个选择条件由case语句来描述

此外select语句对比switch语句来说有诸多的限制 其中最大的一条限制就是每一条语句里面必须有一个IO操作 大致结构如下

	select {case <-chan1: // 如果chan1成功读取到数据 则执行该操作// ....case chan2 <- 1: // 如果chan2成被写入数据 则执行官该操作default:}

在一个select语句中 Go语言会按照顺序评估每个发送和接受的语句 如果说有任意条语句可以执行 那么就从这些可执行的语句中任选一条来使用

如果说所有的通道都被阻塞了 那么此时有两种情况

  • 如果给出了default语句 那么就会执行default语句 并且程序会从select语句后恢复
  • 如果没有default语句 那么default语句将会被阻塞 直到一个case可用
func fib(c, q chan int) {x, y := 1, 1for {select {case c <- x: // 如果c输出了数据x, y = y, x+ycase <-q: // 如果q被写入了数据fmt.Println("quit")return}}
}func main() {c := make(chan int)quit := make(chan int)go func() {for i := 0; i < 6; i++ {fmt.Println(<-c)}quit <- 0}()fib(c, quit)
}

值得注意的是select中 case c <- x: 的含义 它的意思是 c可以写入数据的时候执行 那么c什么时候可以写入数据呢? 当然是有人要接受数据的时候

所以说我们的 fmt.Println(<-c) 语句有两个作用

  1. 接受数据并打印
  2. 让c可以写入数据

运行结果如下

在这里插入图片描述

超时

有时候我们会遇到goroutine阻塞的情况 那么我们如何避免整个程序陷入阻塞呢 我们可以通过设置超时来实现

语法如下

func main() {c := make(chan int)q := make(chan int)o := make(chan bool)go func() {select {case c <- 0: // 当c可以写入数据的时候println("可写入")case <-q: // 当q可以输出数据的时候println("可输出")case <-time.After(5 * time.Second):println("超时")o <- falseprintln("我运行完毕了")break}}()<-o
}

这段代码的最终结果就是打印一个超时之后结束进程

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/213162.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

电商数据采集中如何采集1688平台商品详情SKU数据

一、背景介绍 1688.com是阿里旗下国内最大的B2B批发采购平台&#xff0c;1688分销客是依托此平台的官方营销平台&#xff0c;通过此平台API接口的接入推广平台商家的商品&#xff0c;按照商品成交金额的一定比例获得佣金。可以调用1688平台上的商品详情&#xff0c;SKU数据&…

域控操作五:统一熄屏睡眠时间

直接看图路径&#xff0c;我只设置了熄屏&#xff0c;如果要睡眠就下面那个启用设置时间

本地部署 ComfyUI

本地部署 ComfyUI ComfyUI 介绍ComfyUI Github 地址部署 ComfyUI下载模型启动 ComfyUI访问 ComfyUI ComfyUI 介绍 最强大、模块化的稳定扩散 GUI 和后端。 该用户界面将允许您使用基于图形/节点/流程图的界面设计和执行高级稳定扩散管道。 ComfyUI Github 地址 https://git…

Ubuntu20.04清理垃圾vscode缓存

使用VM虚拟机安装了Ubuntu系统&#xff0c;主目录空间越来越小&#xff0c;硬盘扩容之后很快又空间不足&#xff0c;甚至出现了开机卡黑屏的情况&#xff0c;这里记录一下解决过程。 1 重新开机进入系统 状态&#xff1a;卡到了开机黑屏状态&#xff0c;左上角有一条小横杠 原…

【性能优化】CPU利用率飙高与内存飙高问题

&#x1f4eb;作者简介&#xff1a;小明java问道之路&#xff0c;2022年度博客之星全国TOP3&#xff0c;专注于后端、中间件、计算机底层、架构设计演进与稳定性建设优化&#xff0c;文章内容兼具广度、深度、大厂技术方案&#xff0c;对待技术喜欢推理加验证&#xff0c;就职于…

Mysql 锁机制分析

整体业务代码精简逻辑如下&#xff1a; Transaction public void service(Integer id) {delete(id);insert(id); }数据库实例监控&#xff1a; 当时通过分析上游问题流量限流解决后&#xff0c;后续找时间又重新分析了下问题发生的根本原因&#xff0c;现将其总结如下&#xf…

深入理解路由协议:从概念到实践

路由技术是Internet得以持续运转的关键所在&#xff0c;路由是极其有趣而又复杂的课题&#xff0c;永远的话题。 SO&#xff1a;这是一个解析路由协议的基础文章。 目录 前言路由的概念路由协议的分类数据包在网络中的路由过程理解路由表的结构路由器关键功能解析 前言 在互联…

6.3.WebRTC中的SDP类的结构

在上节课中呢&#xff0c;我向你介绍了sdp协议&#xff0c; 那这节课呢&#xff0c;我们再来看看web rtc中。是如何存储sdp的&#xff1f;也就是sdp的类结构&#xff0c;那在此之前呢&#xff1f;我们先对sdp的内容啊&#xff0c;做一下分类。因为在上节课中呢&#xff0c;虽然…

Redis入门教程

1. 什么是NoSql NoSQL一词最早出现于1998年&#xff0c;是Carlo Strozzi开发的一个轻量、开源、不提供SQL功能的关系数据库。2009年&#xff0c;Last.fm的Johan Oskarsson发起了一次关于分布式开源数据库的讨论&#xff0c;来自Rackspace的Eric Evans再次提出了NoSQL的概念&am…

Python-Django的“日志功能-日志模块(logging模块)-日志输出”的功能详解

01-综述 可以使用Python内置的logging模块来实现Django项目的日志记录。 所以与其说这篇文章在讲Django的“日志功能-日志模块-日志输出”&#xff0c;不如说是在讲Pthon的“日志功能-日志模块-日志输出”&#xff0c;即Python的logging模块。 下面用一个实例来进行讲解。 …

前缀树及其实现解析

前缀树 前缀树&#xff1a;又称单词查找树或键树&#xff0c;是一种哈希树的变种。 典型应用是用于统计和排序大量的字符串&#xff08;但不仅限于字符串&#xff09; 利用字符串的公共前缀来减少查询时间&#xff0c;最大限度地减少无谓的字符串比较。 将一组字符串数组放入…

HTML5+ API 爬坑记录

背景: 有个比较早些使用5开发的项目, 最近两天反馈了一些问题, 解决过程在此记录; 坑1: plus.gallery.pick 选择图片没有进入回调 HTML5 API Reference 在 联想小新 平板电脑上选择相册图片进行上传时, 打开相册瞬间 应用会自动重启, 相册倒是有打开, 不过应用重启了, 导…