day13 Goroutine 协程

news/2024/7/5 2:56:42/文章来源:https://www.cnblogs.com/dengz/p/18279209

了解计算机原理

  • 进程:计算机资源分配单位
  • 线程:cpu处理单位
  • 协程:以 特殊机制或者函数实现高并发,又称 轻量级线程

了解Goroutine

  • Go Goroutine, go语言中的协程,实现并发。
  • 关键字 go
  • 初始大小 4k,随着程序执行自动增长和删除
  • 实现多线程 并发 执行
package mainimport "fmt"func helloGoroutine() {for i := 0; i < 100; i++ {fmt.Println("hello world", i)}
}func main() {/*Go Goroutine, go语言中的协程,实现并发。- 关键字 go- 初始大小 4k,随着程序执行自动增长和删除- 实现多线程 并发 执行*/// 使用 go 实现协程go helloGoroutine()for i := 0; i < 100; i++ {fmt.Println("main ", i)}
}

runtime包

  • 获取系统的信息的工具包
package mainimport ("fmt""runtime""time"
)func goSchedFunc() {go func() {for i := 0; i < 6; i++ {fmt.Println("goroutine", i)}}()runtime.Gosched()for i := 0; i < 6; i++ {fmt.Println("Main goroutine", i)}
}func runtimeGoExitFunc() {// 开启一个 goroutine 线程(协程,轻量级线程)go func() {fmt.Println("Start runtimeGoExitFunc")test()fmt.Println("End runtimeGoExitFunc")}()time.Sleep(time.Second * 3) // 当前函数睡三秒}func test() {defer fmt.Println("延时关闭test函数")runtime.Goexit() // 终止程序,即:结束当前线程(goroutine)fmt.Println("执行test函数")
}func main() {/*runtime 包- 获取系统的信息的工具包*/// 获取 goRoot 目录fmt.Println("GoRoot Path", runtime.GOROOT())// 获取操作系统fmt.Println("System", runtime.GOOS) // darwin// 获取CPU数量fmt.Println("NumCPU", runtime.NumCPU()) // 12// runtime.Gosched() 线程的礼让。让出时间片,即:让goroutine先执行goSchedFunc()// NumGoroutine返回当前存在的运行例程的数量。fmt.Println("NumGoroutine", runtime.NumGoroutine())// runtime.Goexit() 比肩 panic 终止程序runtimeGoExitFunc()//
}

多线程问题

  • 临界资源安全问题。多线程调用共享变量
  • 案例:售票问题
package mainimport ("fmt""time"
)func multithreadedSharedVariableMethods() {shareVariable := 1go func() {shareVariable = 2fmt.Println("GoRoutine shareVariable:", shareVariable)}()shareVariable = 3time.Sleep(1 * time.Second)fmt.Println("main shareVariable:", shareVariable)}
func main() {/*go 中 多线程的问题- 临界资源安全问题。多线程调用共享变量- 案例:售票问题-*/// 案例1。多线程共享变量multithreadedSharedVariableMethods()}

sync 同步锁

  • 锁机制处理多线程抢占资源的问题
    • 某个时间段内,只允许一个 goroutine访问共享数据。当goroutine访问完毕,释放锁后,其他goroutine才能访问
    • 不要以共享内存的方式去通信,而是以通信方式去共享内存
    • 不推荐使用 锁机制。 go语言中! 其他语言基本上都是以锁的方式处理
    • go 中鼓励通过 channel 来解决 共享问题
package mainimport ("fmt""sync""time"
)// SmLock 同步锁【互斥锁】
var SmLock sync.Mutex// Swg 同步等待组。处理子协程执行完成
var Swg sync.WaitGroup// TickerNums 定义全局 票的数量
var TickerNums = 10func saleTicketFunc(ticketSeller string) {// 卖票函数for {if TickerNums > 0 {fmt.Println("获取票总量:", TickerNums)TickerNums--fmt.Println("售票员:", ticketSeller, "卖出1张票", "剩余票的量:", TickerNums)} else {fmt.Println("票卖完了!")break}}
}func saleTicketMainFunc() {// 卖票主函数go saleTicketFunc("张三")go saleTicketFunc("李四")go saleTicketFunc("王五")go saleTicketFunc("赵六")time.Sleep(3 * time.Second)
}func saleTicketMutexWaitGroupFunc(ticketSeller string) {// 卖票函数for {// 增加锁机制,锁住共享资源SmLock.Lock() // 上锁if TickerNums > 0 {fmt.Println("获取票总量:", TickerNums)TickerNums--fmt.Println("售票员:", ticketSeller, "卖出1张票", "剩余票的量:", TickerNums)} else {SmLock.Unlock() // 释放锁,否则进入锁死。fmt.Println("票卖完了!")break}SmLock.Unlock() // 释放锁}
}
func saleTicketMutexWaitGroupMainFunc() {// 增加售票互斥锁,增加上同步等待组。 卖票主函数Swg.Add(4)go saleTicketMutexFunc("张三")go saleTicketMutexFunc("李四")go saleTicketMutexFunc("王五")go saleTicketMutexFunc("赵六")Swg.Wait() // 等待全部协程处理完问题
}func saleTicketMutexFunc(ticketSeller string) {// defer 延时关闭 同步等待组defer Swg.Done()// 卖票函数for {// 增加锁机制,锁住共享资源SmLock.Lock() // 上锁if TickerNums > 0 {fmt.Println("获取票总量:", TickerNums)TickerNums--fmt.Println("售票员:", ticketSeller, "卖出1张票", "剩余票的量:", TickerNums)} else {SmLock.Unlock() // 释放锁,否则进入锁死。fmt.Println("票卖完了!")break}SmLock.Unlock() // 释放锁}
}
func saleTicketMutexMainFunc() {// 增加售票互斥锁。 卖票主函数go saleTicketMutexFunc("张三")go saleTicketMutexFunc("李四")go saleTicketMutexFunc("王五")go saleTicketMutexFunc("赵六")time.Sleep(3 * time.Second)
}func main() {/*sync 包。 锁机制处理多线程抢占资源的问题- 某个时间段内,只允许一个 goroutine访问共享数据。当goroutine访问完毕,释放锁后,其他goroutine才能访问- 不要以共享内存的方式去通信,而是以通信方式去共享内存- 不推荐使用 锁机制。 go语言中! 其他语言基本上都是以锁的方式处理- go 中鼓励通过 channel 来解决 共享问题-*/// 普通卖票。问题:共享资源抢占,sleep处理协程等待执行完成saleTicketMainFunc()// 进阶卖票。处理:解决共享资源抢占问题。 问题:sleep处理协程等待执行完成saleTicketMutexMainFunc()// 高阶卖票。处理:解决共享资源抢占问题,解决协程等待执行完成问题saleTicketMutexWaitGroupMainFunc()}

sync.WaitGroup 同步等待组

package mainimport ("fmt""runtime""sync"
)// SwgTest 定义 同步等待组
var SwgTest sync.WaitGroupfunc test1() {defer SwgTest.Done()for i := 0; i < 10; i++ {fmt.Println("T1", i)}
}
func test2() {for i := 0; i < 10; i++ {fmt.Println("T2", i)}SwgTest.Done()
}func main() {// 同步等待组。处理 goroutine 的等待问题SwgTest.Add(2) // 设定 线程 数量go test1()runtime.Gosched() // 下面的线程让出资源!go test2()// 垃圾处理: time.Sleep(time.Second * 3)//time.Sleep(time.Second * 1)// 优秀的处理: 同步等待组等待执行完成SwgTest.Wait()}

sync 的 锁 Lock + WaitGroup 同步等待组

package mainimport ("fmt""sync"
)// TickerNums2 定义全局 票的数量
var TickerNums2 = 10// SmLock2 同步锁【互斥锁】
var SmLock2 sync.Mutex// Swg2 同步等待组。处理子协程执行完成
var Swg2 sync.WaitGroupfunc saleTicketMutexFunc2(ticketSeller string) {// defer 延时关闭 同步等待组defer Swg2.Done()// 卖票函数for {// 增加锁机制,锁住共享资源SmLock2.Lock() // 上锁if TickerNums2 > 0 {fmt.Println("获取票总量:", TickerNums2)TickerNums2--fmt.Println("售票员:", ticketSeller, "卖出1张票", "剩余票的量:", TickerNums2)} else {SmLock2.Unlock() // 释放锁,否则进入锁死。fmt.Println("票卖完了!")break}SmLock2.Unlock() // 释放锁}
}
func saleTicketMutexWaitGroupMainFunc2() {// 增加售票互斥锁,增加上同步等待组。 卖票主函数Swg2.Add(4)go saleTicketMutexFunc2("张三")go saleTicketMutexFunc2("李四")go saleTicketMutexFunc2("王五")go saleTicketMutexFunc2("赵六")Swg2.Wait() // 等待全部协程处理完问题
}
func main() {// 高阶卖票。处理:解决共享资源抢占问题,解决协程等待执行完成问题saleTicketMutexWaitGroupMainFunc2()
}

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

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

相关文章

camunda多租户技术架构介绍和测试验证

多租户考虑的是单个 Camunda 安装应该为多个租户提供服务的情况。对于每个租户,应做出一定的隔离保证。例如,一个租户的流程实例不应干扰另一租户的流程实例。 多租户可以通过两种不同的方式实现。一种方法是每个租户使用一个流程引擎。另一种方法是仅使用一个流程引擎并将数…

IEEE 8802-3 以太网标准解读

PHY: CarrierSense 载波侦听 ReceveDataValid 接受数据有效 CollisionDetect 碰撞检测 Transmitting 传输中 TransmitBit 传输比特 SFD 10101011 开始 ReceiveBit 接受比特 Wait 等待1、MA_DATA.request 定义了MAC客户端访问单独实体或者z在组地址的前提下访问多个实体 MA…

在C#中使用RabbitMQ做个简单的发送邮件小项目

在C#中使用RabbitMQ做个简单的发送邮件小项目 前言 好久没有做项目了,这次做一个发送邮件的小项目。发邮件是一个比较耗时的操作,之前在我的个人博客里面回复评论和友链申请是会通过发送邮件来通知对方的,不过当时只是简单的进行了异步操作。 那么这次来使用RabbitMQ去统一发…

【进阶篇】Java 项目中对使用递归的理解分享

笔者在最近的项目开发中,遇到了两个父子关系紧密相关的场景:评论树结构、部门树结构。具体的需求如:找出某条评论下的所有子评论id集合,找出某个部门下所有的子部门id集合。【进阶篇】Java 项目中对使用递归的理解分享 目录【进阶篇】Java 项目中对使用递归的理解分享前言一…

C#/.NET/.NET Core编程技巧练习集(学习,实践干货)

DotNet Exercises介绍 DotNetGuide专栏C#/.NET/.NET Core编程常用语法、算法、技巧、中间件、类库练习集,配套详细的文章教程讲解,助你快速掌握C#/.NET/.NET Core各种编程常用语法、算法、技巧、中间件、类库等等。GitHub开源地址:https://github.com/YSGStudyHards/DotNetE…

2024年6月文章一览

2024年6月编程人总共更新了5篇文章: 1.2024年5月文章一览 2.《编译原理》阅读笔记:p18 3.《编译原理》阅读笔记:p19-p24 4.《编译原理》阅读笔记:p25-p32 5.《Programming from the Ground Up》阅读笔记:p1-p18 6月再挖一个坑,开始《Programming from the Ground Up》的学…

【论文阅读】自动驾驶光流任务 DeFlow: Decoder of Scene Flow Network in Autonomous Driving

再一次轮到讲自己的paper!耶,宣传一下自己的工作,顺便完成中文博客的解读 方便大家讨论。Title PictureReference and pictures paper: https://arxiv.org/abs/2401.16122 code: https://github.com/KTH-RPL/DeFlow b站视频: https://www.bilibili.com/video/BV1GH4y1w7LQ1.…

利用SonarCloud和Azure DevOps提升代码质量

利用SonarCloud和Azure DevOps提升代码质量 在软件开发过程中,代码质量是至关重要的。为了确保代码的清洁和安全性,我们可以使用SonarCloud——一个基于云的代码质量与安全服务。SonarCloud不仅对开源项目免费,还为私有项目提供了14天的免费试用。本文将指导您如何将SonarCl…

免费申请一张SSL证书-包含100个域名-挑战!!

挑战一下,申请一张包含100个域名的证书 首先,我们访问来此加密网站,进入登录页面,输入我的账号密码。登录后,咱们就可以开始申请证书,首先说一下,咱账号是SVIP哦,只有SVIP才可以申请包含100个域名的证书。为了方便验证,咱么先添加一下域名接口,这样可以自动配置验证信…

.net6+ 在单文件应用程序中获取程序集位置

一般来说,获取执行程序集的位置,您可以调用: var executableDirectory = System.Reflection.Assembly.GetExecutingAssembly().Location;如果发布为单个文件, 会提示如下警告 warning IL3000: System.Reflection.Assembly.Location always returns an empty string for assemb…

(set+拓扑排序) CF1572A Book

题意:思路: 每本书之间很明显存在拓扑关系,由此想到拓扑排序。使用set对图进行拓扑排序,将阅读次数小的放在前面,若阅读次数相同则按照阅读章节编号排序。假设第 x 章在第 y 章理解之后就能理解,若 x 大于 y 则本次阅读就可以理解 x 章,否则需要下一次才能理解第 x 章。 …

【github报错解决】Failed to connect to github.com port 443: Couldnt connect to server

今天使用github push时候报错的 解决办法: 1、查看本机代理 路径:设置->网络和Internet->代理->地址:端口 2、用git cmd修改配置,port就用刚刚代理端口值git config --global http.proxy http://127.0.0.1:[port] git config --global https.proxy http://127.0.0…

LVGL组件

LVGL组件的使用 目录LVGL组件的使用1 . 父和子对象2. 部件的基本属性及设置3. 图解:还有其他样式可以设置2. 组件分类 1 . 父和子对象2. 部件的基本属性及设置 /******** 设置打小 ********/ /******** 注意:设置部件位置时,坐标原点在父对象的左上角 ********/ lv_…

Maven 笔记

开发工具:Maven相关内容笔记# Maven简介 Maven的本质是一个项目管理工具,将项目开发和管理过程抽象成一个项目对象模型(POM)) 这玩意儿是使用Java开发的,所以采用的就是Java的思想:面向对象 POM (Project Object Model):项目对象模型Maven的作用:项目构建:提供标准的、…

202406月度小结

这个学期比上个学期有意思得多了。但是为什么我只写了6月小结,没有写3月4月5月小结呢?因为那时候还没有想到用博客写这种小结形式的随笔啦…… 横向对比,这四个月都挺有意思的。但是由于我太容易忘记事情了……我的记忆是有限的,先把6月发生的趣事赶紧记录下来比较好。 昨天…

大模型技术方向Task1笔记

赛题概要 一、赛事背景 在当今数字化时代,企业积累了丰富的对话数据,这些数据不仅是客户与企业之间交流的记录,更是隐藏着宝贵信息的宝库。在这个背景下,群聊对话分角色要素提取成为了企业营销和服务的一项重要策略。 群聊对话分角色要素提取的理念是基于企业对话数据的深度…

[LeetCode] 122. Best Time to Buy and Sell Stock II

medium是你的谎言. class Solution:def maxProfit(self, prices: List[int]) -> int:#1if len(prices) == 1:return 0#elsemax_profit = 0min_price = prices[0]for i, element in enumerate(prices):#find a min_price if element <= min_price:min_price = elementelse…

【esp32 学习笔记】esp-idf学会调用组件管理——以button

简单不看版——esp-idf组件管理步骤 在ESP-IDF 组件管理器网页(https://components.espressif.com/)搜索我们需要的组件,比如【button】,然后 点开相应的组件,比如 espressif/button 组件。 【关键步骤】复制相关组件界面上配置组件的命令 ,形如:idf.py add-dependency &…

idea创建javaweb项目

1.新建project项目 2.添加依赖<dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>6.0.10</version> </dependency>…

Linguistics-English-Would, Should, and Could: How to Use Them Correctly

https://7esl.com/would-should-could/ Key Takeaways“Would” is used for hypotheticals and future possibilities that may not occur. “Should” implies advice, expectation, or probability. “Could” expresses past ability or present possibility.Home Knowled…