Go并发编程(上)

news2025/1/10 21:28:14

目录

一、go语言当中的协程

二、MPG模型介绍

三、Goroutine  的使用

 3.1  协程的开启

3.2 优雅地等待子协程结束

四、捕获子协程的panic

五、管道Channel

5.1、认识管道

5.2、Channel的遍历和关闭

5.3 、用管道实现生产者消费者模型

5.4、Channel一些使用细节和注意事项


一、go语言当中的协程

在C++中我们要实现并发编程的时候,我们通常需要自己维护一个线程池,并且需要自己去包装一个又一个的任务,同时需要自己去调度线程执行任务并维护上下文切换,这一切通常会耗费程序员大量的心智。那么能不能有一种机制,程序员只需要定义很多个任务,让系统去帮助我们把这些任务分配到CPU上实现并发执行呢?

Go语言中的goroutine就是这样一种机制,goroutine的概念类似于线程,但 goroutine是由Go的运行时(runtime)调度和管理的。Go程序会智能地将 goroutine 中的任务合理地分配给每个CPU。Go语言之所以被称为现代化的编程语言,就是因为它在语言层面已经内置了调度和上下文切换的机制


Go语言编程中你不需要去自己写进程、线程、协程,你的技能包里只有一个技能–goroutine,当你需要让某个任务并发执行的时候,你只需要把这个任务包装成一个函数,开启一个goroutine去执行这个函数就可以了,就是这么简单粗暴

二、MPG模型介绍

我们先来看一下Go语言的并发模式,发现是不同于C++的

下面我们来解释一下MPG模式当中的M、P、G分别代表什么意思

模型解释:

G0, G1,G2  谁先执行完是完全不确定的,这不像是java语言,java可以给每个线程分别设置一个优先级,然后控制线程的执行顺序,但是go的话是不行的。  程序员只能把一个协程开启,但是中间的过程是无法去决定的。

三、Goroutine  的使用

 3.1  协程的开启

两种方法开启,当然这里只是开启,并没有去等待协程的结束。

3.2 优雅地等待子协程结束

父协程结束后,子协程并不会结束。main协程结束后,所有协程都会结束。

代码演示:

var wg = sync.WaitGroup{}

func Add() {
	defer wg.Done()
	time.Sleep(1 * time.Second)
	fmt.Println("over")
}

func main() {
	wg.Add(2)
	go Add() //开启了一个协程,并没有等待结束
	go Add()
	wg.Wait()
}

四、捕获子协程的panic

何时会发生panic:

  • 运行时错误会导致panic,比如数组越界、除0。

  • 程序主动调用panic(error)。

panic会执行什么:

  1. 逆序执行当前goroutine的defer链(recover从这里介入)。

  2. 打印错误信息和调用堆栈。

  3. 调用exit(2)结束整个进程。

关于defer

  • defer在函数退出前被调用,注意不是在代码的return语句之前执行,因为return语句不是原子操作。

  • 如果发生panic,则之后注册的defer不会执行。

  • defer服从先进后出原则,即一个函数里如果注册了多个defer,则按注册的逆序执行。

  • defer后面可以跟一个匿名函数。

五、管道Channel

5.1、认识管道

管道其本质上是一个环形队列,在这里说明一下定义管道有以下节点需要注意

1.hannel本质就是一个数据结构-环形队列
2.数据是先进先出[FIFO : [first in first out]
3.线程安全,多goroutine访问时,不需要加锁,就是说channel本身就是线程安全的(编译器底部维护的)

4.channel有类型的,一个string的channel只能存放string类型数据

下面我们来看看如何定义管道

var intChan chan int //intChan用来存储int数据
	var mapChan chan map[int]string//mapChan用来存储map[int]string类型
	var perChan chan People//用来存储自定义类型People
	var perChan2 chan *People

在这里有以下几点需要注意

  • 管道channel 是引用类型,需要初始化以后才能插入数据,也就是make
  • 管道是有类型的,管道的类型是什么就只能写入这种类型的数据
  • 当管道写满了以后,在没有别的协程的情况下,再次写入会导致死锁
  • 在没有使用协程的情况下(取完没放入),当管道为空,再取,会报deadlock
  • 遍历管道时需要提前把管道关闭(close),否则会导致死锁

代码进行演示

func main() {
	ch := make(chan int, 5)
	//创建一个管道
	for i := 0; i < 5; i++ {
		ch <- i //在管道当中写入数据
	}
	for len(ch) > 0 {
		value := <-ch
		fmt.Println(value)
	}

	ch = make(chan int) //非缓冲通道
	val := <-ch         //没有初始化就取数据,会发生错误的
	fmt.Println(val)
	//注意channel关闭之后不能向channel当中写入数据,否则会造成死锁
	/*
	   channel支持for --range遍历但是请注意两个细节
	   遍历时如果channel没有关闭则会出现deadlock的错误
	   在遍历时如果channel以及关闭了则会正常的遍历数据,遍历完毕之后就会退出吧遍历
	*/

	intchan := make(chan int, 100)
	for i := 0; i < 100; i++ {
		intchan <- i
		//放入100个数据到channel当中
	}

	//for i := 0; i < len(intchan); i++ {
	//	fmt.Println(<-intchan)
	//	//注意这样会少50个数据所以不能这样遍历管道的长度是一直在变的
	//}

	close(intchan) //遍历其一定需要将管道关闭否则会造成死锁
	for v := range intchan {
		fmt.Println(v)
	}
}

5.2、Channel的遍历和关闭

1.channel的关闭:使用内置函数close可以关闭channel,当channel关闭后,就不能再向channel写数据了,但是仍然可以从该channel读取数据,就算读取数据个数大于容量,也能读取到,只是读出来的是0 ,这时候就需要用  v,ok := < - intChan,  如果管道没有关闭,是会阻塞在这一步的,也就是说ok这里不会有值

for {
		v, ok := <-intChan2
		if !ok {//证明没有数据了
			fmt.Println("没有数据了") 
			fmt.Println(v) //  读出来的是0
			break
		}
		fmt.Println(v)
	}

2.channel支持for-range 遍历和普通for进行遍历,但是普通的for循环遍历, 因为取出操作本身会导致长度变化所以我们不建议使用。

3.在遍历时,如果channel没有关闭,则回出现deadlock的错误。在遍历时,如果channel已经关闭,则会正常遍历数据,遍历完后,就会退出遍历。
 

func main() {
	intChan := make(chan int, 3)
	intChan <- 1
	intChan <- 2

	close(intChan) //关闭管道
	//关闭管道后就不能存放了,但是可以取数据
	x1 := <-intChan
	x2 := <-intChan
	//x3 := <-intChan
	//x4 := <-intChan

	fmt.Println(x1)
	fmt.Println(x2)
	//fmt.Println(x3)
	//fmt.Println(x4)

	intChan2 := make(chan int, 100)
	for i := 0; i < 100; i++ {
		intChan2 <- i
	}
	//遍历管道前切记要先close
	//不能使用普通的for循环,因为管道的数量在动态变化的 ,但是这里如果提前直到数量是100,循环的话i<100 就行
	//for i := 0; i < len(intChan2); i++ {
	//	x := <-intChan2
	//	fmt.Println(x)
	//}

	close(intChan2) //遍历前切记要先关闭管道
	for v := range intChan2 {
		fmt.Println(v)
	}

}

5.3 、用管道实现生产者消费者模型

1.开启一个Writea协程,向管道intChan中写入50个整数
2.开启一个Read协程,从管道intChan中读取writeData写入的数据。

3.注意: Write和Read操作的是同一个管道
4.主线程需要等待Write和Read协程都完成工作才能退出[管道]

代码展示:

package main

import (
	"fmt"
)

func Write(intChan chan int) {
	for i := 0; i < 50; i++ {
		fmt.Printf("写入数据 :%d \n", i)
		intChan <- i
	}
	close(intChan) //写完之后关闭管道,但是不影响取
}

func Read(intChan chan int, boolChan chan bool) {
	for {
		//time.Sleep(time.Millisecond * 100) //先等一秒,为了留时间给生产者,不等可能会直接退出
		v, ok := <-intChan
		if !ok { //证明已经没有数据了
			fmt.Println("管道已经没有数据了。。。")
			break
		}
		fmt.Printf("读出了数据 :%d\n", v)
	}
	//任务完成向管道当中写入数据
	boolChan <- true
	//通知主线程该结束了
	close(boolChan)
}

func main() {
	intChan := make(chan int, 10)
	boolChan := make(chan bool, 1) //用来标记是否执行完的管道

	go Read(intChan, boolChan) //开启消费者
	go Write(intChan)          //开启生产者

	for { //让主线程一直在这里等待,直到boolChan有数据了
		_, ok := <-boolChan
		if ok {
			break
		}
	}
}

5.4、Channel一些使用细节和注意事项

在go语言当中如果某个协程出现了异常,如果我们不做任何处理那么就会导致整个程序崩溃掉。在go语言当中我们可以使用   

     defer + recover来处理整个异常。

如果我们起了一个协程,但是这个协程出现了panic,如果我们没有捕获这个panic,就会造成整个程序崩溃,这时我们可以在goroutine中使用recover来捕获panic,进行处理,这样即使这个协程发生的问题,但是主线程仍然不受影响,可以继续执行

package main

import (
	"fmt"
	"time"
)

/*
   go语言当中使用recover解决协程当中出现的panic导致程序崩溃的问题
   如果一个协程出现了异常会导致整个程序崩溃,此时我们需要使用recover来捕获这个panic这样就不会影响其它协程

*/

func Say() {
	for i := 0; i < 10; i++ {
		fmt.Println("hello world")
	}
}

func Test() {
	//使用defer + recover捕获抛出的panic
	defer func() { 
		if err := recover(); err != nil {
			fmt.Println("test()协程发生错误:\n", err)
		}
	}()
	var myMap map[int]string //需要提前make
	myMap[0] = "提升和"         //没有提前make

}
func main() {

	go Test()
	go Say()

	time.Sleep(time.Second)
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

揭秘!2024年热门的图标设计工具

在这个瞬息万变的世界里&#xff0c;设计师们对创新和实用的工具的渴望日益热切。我们需要时刻紧跟潮流&#xff0c;发掘和尝试最新&#xff0c;最有价值的图标设计工具&#xff0c;才能在剧烈的市场竞争中引人注目。以下是我们细心挑选的2024年图标设计工具的热门推荐&#xf…

CentOS 7 双网卡绑定热备 —— 筑梦之路

为什么需要&#xff1f; 1. 增强网络的可靠性 2. 保障服务的可持续性 3. 降低网卡故障带来的不良影响 有哪些模式&#xff1f; 模式0&#xff1a;轮询策略&#xff08;round robin&#xff09;&#xff0c;mode0&#xff0c;优点&#xff1a;流量提高一倍缺点&#xff1a;需要接…

《QT从基础到进阶·十五》用鼠标绘制矩形(QGraphicsView、QPainter、QGraphicsRectItem)

以下是鼠标绘制矩形最全的一种用法&#xff0c;完整源码将会放在最后面。 QT版本&#xff1a;5.15.2 VS版本&#xff1a;2019 1、在界面加载一张图片 界面的搭建选用QGraphicsView&#xff0c;自定义类GraphicsView继承QGraphicsView&#xff0c;在主程序中点击按钮打开 图片&…

opencv4笔记

图像二值化 全局法Threshold 大津法 大津法OSTU阈值类型——适用于双峰直方图 OTSU算法也称最大类间差法&#xff0c;由大津于1979年提出&#xff0c;被认为是图像分割中阈值选取的最佳算法&#xff0c;计算简单&#xff0c;不受图像亮度和对比度的影响&#xff0c;它是按图…

浅谈泛在电力物联网在智能配电系统应用

贾丽丽 安科瑞电气股份有限公司 上海嘉定 201801 摘要&#xff1a;在社会经济和科学技术不断发展中&#xff0c;配电网实现了角色转变&#xff0c;传统的单向供电服务形式已经被双向能流服务形式取代&#xff0c;社会多样化的用电需求也得以有效满足。随着物联网技术的发展&am…

c#数据类型

常量 /*常量是固定的量&#xff0c;在运行过程中不可以改变的量 const 来修饰不能改变的量*/ // public private protected internal 是访问修饰符using System.Security.Cryptography.X509Certificates;namespace ConsoleApp1 {internal class Program{public const int a 1…

科技云报道:数智化升级,如何跨越数字世界与实体产业的鸿沟?

科技云报道原创。 数智化是当下商业环境下最大的确定性。 2022年&#xff0c;中国数字经济规模达50.2万亿元&#xff0c;占国内生产总值比重提升至41.5%&#xff0c;数字经济成为推动经济发展的重要引擎。从小型创业公司到跨国巨头&#xff0c;数字化转型在企业发展历程中彰显…

javaSE学习笔记(二)数组,类,对象,成员变量,匿名对象,构造方法,static,final,封装,继承,多态

目录 三、面向对象 1.概述 面向过程与面向对象 面向对象编程特点 面向对象三个基本特征 2.数组 数组定义格式 数组的初始化 动态初始化 静态初始化 数组的内存分配 Java中的内存分配 数组的内存分配 数组的角标 数组的基本操作 二维数组&#xff08;实际开发几乎…

maven 下载和安装

点击进入Maven下载网址 Maven – Download Apache Maven Maven详细下载列表 Index of /dist/maven/maven-3 本地仓库配置文件 配置环境变量 idea编辑器配置 maven Javajdk配置 字节码版本是否11

yolov5 通过视频进行目标检测

打开yolov5-master文件夹&#xff0c;可以看到一个名为data的文件夹&#xff0c;在data中创建一个新的文件夹&#xff0c;命名为videos。 打开yolov5-master中的detect.py可以看到一行代码&#xff08;大概在245行左右&#xff09;为 parser.add_argument(--source, typestr,…

POI.5.2.4常用操作-数据导入导出常规操作

1、APACHE POI简介 Apache POI 简介是用Java编写的免费开源的跨平台的 Java API&#xff0c;Apache POI提供API给Java程式对Microsoft Office&#xff08;Excel、WORD、PowerPoint、Visio等&#xff09;格式档案读和写的功能。 1.1、其他操作Excel工具 Java Excel是一开放源码…

类直径树上贪心

http://cplusoj.com/d/senior/p/SS231109C 场上想到枚举点&#xff0c;然后最大值为高&#xff0c;然后可以求最大值。但是感觉计数会重 计数其实不会重&#xff0c;如图中&#xff0c;红色线段显然比蓝色线段优 所以我们枚举3叉点时没错的 #include<bits/stdc.h> usin…

Terminator终端

Terminator终端 terminator快捷键 sudo apt install terminator使用ctrlaltt开启终端 terminator快捷键 E O W 都是ctrl shift结合 alt方向键&#xff0c;为切换焦点 e分屏失效的解决方案&#xff08;原因是快捷键占有问题&#xff09; ibus-setup将表情符号注释就行了 …

ARMday2(环境创建+工程配置+创建文件+单步调试)

目录 一、汇编环境的创建 二、为工程配置链接脚本&#xff08;map.lds&#xff09; 三、为工程创建汇编文件 start.s 编程调试 接下来我们需要建立一个 start.s 汇编文件添加到我们的工程中去 四、对汇编代码进行单步调试&#xff08;仿真&#xff09; 五、汇编工程的编译 …

遍历List集合和Map进行修改和删除报java.util.ConcurrentModificationException错误详解

一、异常产生 当我们使用foreach迭代一个ArrayList或者HashMap时&#xff0c;如果尝试对集合做一些修改操作&#xff08;例如删除元素或新增&#xff09;&#xff0c;可能会抛出java.util.ConcurrentModificationException的异常。 javapublic static void main(String[] args)…

语音控制:基于ESP8266的DIY助手

随着智能家居的兴起&#xff0c;语音控制成为越来越受欢迎的方式。在本篇文章中&#xff0c;我们将向您介绍如何使用ESP8266构建一个基于语音控制的DIY助手。 第一步&#xff1a;硬件准备 在开始之前&#xff0c;您需要准备一些基础硬件&#xff0c;包括ESP8266模块、麦克风模…

【C++】C++中的IO流

文章目录 一、C语言的输入与输出二、什么是流三、CIO流1.C标准IO流2.C文件IO流 四、stringstream的简单介绍 一、C语言的输入与输出 C语言中我们用到的最频繁的输入输出方式就是scanf ()与printf(): scanf(): 从标准输入设备(键盘)读取数据&#xff0c;并将值存放在变量对应的…

数据结构-图的存储

邻接矩阵法 #define MaxVertexNum 100 //顶点数目的最大值 typedef struct{char Vex[MaxVertexNum]; //顶点表int Edge[MaxVertexNum][MaxVertexNum]; //邻接矩阵&#xff0c;边表int vexnum,arcnum; //图的当前顶点数和边数 }MGraph;无向图 第i个顶点的度第i行&#xff08;或第…

深入理解JVM虚拟机第二十二篇:详解JVM当中与操作数栈相关的字节码指令

大神链接&#xff1a;作者有幸结识技术大神孙哥为好友&#xff0c;获益匪浅。现在把孙哥视频分享给大家。 孙哥链接&#xff1a;孙哥个人主页 作者简介&#xff1a;一个颜值99分&#xff0c;只比孙哥差一点的程序员 本专栏简介&#xff1a;话不多说&#xff0c;让我们一起干翻J…

VUE页面导出PDF方案

1&#xff0c;技术方案为&#xff1a;html2canvas把页面生成canvas图片&#xff0c;再通过jspdf生成PDF文件&#xff1b; 2&#xff0c;安装依赖&#xff1a; npm i html2canvas -S npm i jspdf -S 3&#xff0c;封装导出pdf方法exportPdf.js: // 页面导出为pdf格式 //titl…