GO语言:Worker Pools线程池、Select语句、Metex互斥锁详细示例教程

目录标题

  • 一、Buffered Channels and Worker Pools
    • 1. Goroutine and Channel Example 线程和通道示例
    • 2. Deadlock 死锁
    • 3. Closing buffered channels 关闭通道
    • 4. Length vs Capacity 长度和容量
    • 5. WaitGroup
    • 6. Worker Pool Implementation 线程池
  • 二、Select
    • 1. Example
    • 2. Default case 默认选择
    • 3. Deadlock and default case 死锁与默认选择
    • 4. Random selection 随机选择
  • 三、Mutex
    • 1. Program with a race condition 无锁示例
    • 2. Solving the race condition using a mutex 互斥锁解决方案
    • 3. Solving the race condition using channel 通道解决方案

一、Buffered Channels and Worker Pools

1. Goroutine and Channel Example 线程和通道示例

          package mainimport ("fmt""time")func write(ch chan int) {for i := 0; i < 5; i++ {ch <- i // 向通道写入 0-4		因为通道容量是2 需要读取数据才会进行下一步 否则一直在阻塞态fmt.Println("Successfully wrote", i, "to ch")}close(ch) // 关闭通道}func main() {ch := make(chan int, 2) // 创建一个容量为2的缓冲通道  通道容量大小会导致阻塞go write(ch)time.Sleep(2 * time.Second) // 模拟时间间隔for v := range ch {fmt.Println("Read value", v, "from ch") // 读取数据 goroutine继续运行time.Sleep(2 * time.Second)}// 并发的 goroutine 和通道的阻塞机制,write() 函数和 range ch 循环可以交替执行,使得循环不会一次执行完毕,而是在读取完所有值之后等待新的值出现,再次进行循环迭代。}// Successfully wrote 0 to ch// Successfully wrote 1 to ch// Read value 0 from ch// Successfully wrote 2 to ch// Read value 1 from ch// Successfully wrote 3 to ch// Read value 2 from ch// Successfully wrote 4 to ch// Read value 3 from ch// Read value 4 from ch

2. Deadlock 死锁

         package mainimport (  "fmt")func main() {  ch := make(chan string, 2)ch <- "naveen"ch <- "paul"ch <- "steve"			// 其容量是2 但是写入三个 导致死锁fmt.Println(<-ch)fmt.Println(<-ch)}// fatal error: all goroutines are asleep - deadlock!// goroutine 1 [chan send]:  // main.main()  //     /tmp/sandbox091448810/prog.go:11 +0x8d

3. Closing buffered channels 关闭通道

         ch := make(chan int, 5)ch <- 6ch <- 9close(ch)n, open := <-chfmt.Printf("Received: %d, open: %t\n", n, open)n, open = <-chfmt.Printf("Received: %d, open: %t\n", n, open)n, open = <-chfmt.Printf("Received: %d, open: %t\n", n, open)// Received: 5, open: true  // Received: 6, open: true  // Received: 0, open: false  

4. Length vs Capacity 长度和容量

        ch := make(chan string, 3)ch <- "Like"ch <- "LiangXiaoQing"fmt.Println("capacity is", cap(ch))fmt.Println("length is", len(ch)) // 通道写入的个数fmt.Println("read value", <-ch)fmt.Println("new length is", len(ch))fmt.Println("read value", <-ch)fmt.Println("new length is", len(ch))// capacity is 3// length is 2// read value Like// new length is 1// read value LiangXiaoQing// new length is 0

5. WaitGroup

         // Add() 添加任务// Done() 通知wait完成任务// Wait() 阻塞等待所有任务完成package mainimport (  "fmt""sync""time")func process(i int, wg *sync.WaitGroup) {fmt.Println("started Goroutine", i)   // 3.打印 Goroutine 开始执行的信息time.Sleep(2 * time.Second)           // 4.暂停 2 秒,模拟任务执行时间fmt.Printf("Goroutine %d ended\n", i) // 5.打印 Goroutine 执行结束的信息wg.Done()                             // 6.通知等待组任务已完成}func main() {no := 3var wg sync.WaitGroupfor i := 0; i < no; i++ {wg.Add(1)          // 1.循环三次添加三次任务go process(i, &wg) // 2.每次传入当前i 0-2 及wg内存地址}wg.Wait() 	// 7.等待所有任务完成fmt.Println("All go routines finished executing")}// started Goroutine 1// started Goroutine 0// started Goroutine 2// Goroutine 2 ended// Goroutine 0 ended// Goroutine 1 ended// All go routines finished executing

6. Worker Pool Implementation 线程池

        type Job struct { // Job 结构表示一个具有 ID 和随机数的作业。id       intrandomno int}type Result struct { // Result 结构表示作业的结果,包括作业本身和数字各位数之和。job         Jobsumofdigits int}var jobs = make(chan Job, 10)       // jobs 是一个带有缓冲区大小为 10 的通道,用于传递作业。var results = make(chan Result, 10) // results 是一个带有缓冲区大小为 10 的通道,用于传递结果。func digits(number int) int { // digits 函数计算一个整数的各位数之和。sum := 0no := numberfor no != 0 { // 循环中,通过取模和除法操作,将数字的各位数相加。digit := no % 10sum += digitno /= 10}time.Sleep(2 * time.Second) // time.Sleep(2 * time.Second) 使函数暂停 2 秒钟,模拟一个耗时操作。return sum                  // 返回各位数之和。}func worker(wg *sync.WaitGroup) { // worker 函数是一个工作协程,用于处理作业。for job := range jobs { // 使用 range 循环从 jobs 通道接收作业。output := Result{job, digits(job.randomno)} // 通过调用 digits 函数计算作业的各位数之和。 将作业和结果封装为 Result 结构results <- output                           // 并发送到 results 通道。}wg.Done() // wg.Done() 声明一个任务已完成。}func createWorkerPool(noOfWorkers int) { // createWorkerPool 函数创建一个工作池,用于并发处理作业。var wg sync.WaitGroup              // 创建一个 sync.WaitGroup 对象 wg,用于等待所有工作协程完成。for i := 0; i < noOfWorkers; i++ { // 使用 for 循环创建指定数量的工作协程。wg.Add(1)      // 添加任务go worker(&wg) // 每个工作协程调用 worker 函数,并传递 &wg 作为参数。}wg.Wait()      // wg.Wait() 等待所有工作协程完成。close(results) // 关闭 results 通道,表示所有结果已经发送完毕。}func allocate(noOfJobs int) { // allocate 函数用于生成指定数量的作业并发送到 jobs 通道。for i := 0; i < noOfJobs; i++ { //  使用 for 循环创建指定数量的作业。randomno := rand.Intn(999) // 生成一个随机数 randomno,范围在 0 到 999 之间。job := Job{i, randomno}    // 创建一个 Job 结构体 jobjobs <- job                // 并将其发送到 jobs 通道。}close(jobs) // 关闭 jobs 通道,表示所有作业已经发送完毕。}func result(done chan bool) { // result 函数用于接收并处理结果。for result := range results { // 使用 range 循环从 results 通道接收结果。fmt.Printf("Job id %d, input random no %d , sum of digits %d\n", result.job.id, result.job.randomno, result.sumofdigits)}done <- true}func main() {startTime := time.Now()noOfJobs := 10go allocate(noOfJobs) 	// 传入job id 0-99 random 0-999的随机数字	Job id 1, input random no 636done := make(chan bool)go result(done)		// result线程一直等待results有值 如果有值则打印信息 传入true结束通道noOfWorkers := 10	// 控制线程池数量 createWorkerPool(noOfWorkers)	// 把&wg sync.WaitGroup 类型变量的指针 传给 results<-done	// 关闭通道endTime := time.Now()diff := endTime.Sub(startTime)	// 计算时间差fmt.Println("total time taken", diff.Seconds(), "seconds")}// Job id 1, input random no 636, sum of digits 15  // Job id 0, input random no 878, sum of digits 23  // Job id 9, input random no 150, sum of digits 6  // ...// total time taken  20.01081009 seconds  

二、Select

1. Example

        package mainimport ("fmt""time")func server1(ch chan string) {time.Sleep(9 * time.Second)ch <- "From Server 1"}func server2(ch chan string) {time.Sleep(6 * time.Second)ch <- "From server 2"}func main() {output1 := make(chan string)output2 := make(chan string)go server1(output1)go server2(output2)select { 	// 使用select语句接收多个通道消息,select会接收最先准备好的通道接收操作case s1 := <-output1:fmt.Println(s1)case s2 := <-output2:fmt.Println(s2)}}// From server 2

2. Default case 默认选择

        func process(ch chan string) {time.Sleep(1 * time.Second)ch <- "Process Successful"}func main() {ch := make(chan string)go process(ch)for { 		// for循环一直循环 每次循环休息1秒 直到v有值 主要看上面process函数睡眠睡觉 否则一直输出default的值time.Sleep(1000 * time.Microsecond)select {case v := <-ch:fmt.Println("Received value:", v)returndefault:fmt.Println("No value Received")}}// ....// No value Received//No value Received//No value Received//No value Received//Received value: Process Successful

3. Deadlock and default case 死锁与默认选择

        func main() {ch := make(chan string)select {case v := <-ch:fmt.Println("Received value", v)default:fmt.Println("Default case executed")}}// Default case executed

4. Random selection 随机选择

        package mainimport (  "fmt""time")func server1(ch chan string) {  ch <- "from server1"}func server2(ch chan string) {  ch <- "from server2"}func main() {  output1 := make(chan string)output2 := make(chan string)go server1(output1)go server2(output2)time.Sleep(1 * time.Second)select {		// 使用select语句接收多个通道消息,select会接收最先准备好的通道接收操作case s1 := <-output1:fmt.Println(s1)case s2 := <-output2:fmt.Println(s2)}}// From Server 1

三、Mutex

1. Program with a race condition 无锁示例

        package mainimport ("fmt""sync")var x = 0func increment(wg *sync.WaitGroup) {x = x + 1wg.Done()}func main() {var w sync.WaitGroupfor i := 0; i < 1000; i++ {w.Add(1)go increment(&w)}w.Wait()fmt.Println("Final value of X", x)}// Final value of X 987		最终答案应该是1000 因为多线程全部都在操作x 导致有些操作未成功

2. Solving the race condition using a mutex 互斥锁解决方案

        package mainimport ("fmt""sync")var x = 0func increment(wg *sync.WaitGroup, m *sync.Mutex) {m.Lock() // 上锁x = x + 1m.Unlock() // 释放锁  只有拿到锁才能操作x 否则一直等待wg.Done()}func main() {var w sync.WaitGroupvar m sync.Mutexfor i := 0; i < 1000; i++ {w.Add(1)go increment(&w, &m)}w.Wait()fmt.Println("Final value of X", x)}// Final value of X 1000

3. Solving the race condition using channel 通道解决方案

        package main  import (  "fmt""sync")var x  = 0  func increment(wg *sync.WaitGroup, ch chan bool) {  ch <- truex = x + 1<- chwg.Done()   }func main() {  var w sync.WaitGroupch := make(chan bool, 1)	// 通道容量 1	所以每次都需要上一个结束 下一个才能进行操作for i := 0; i < 1000; i++ {w.Add(1)        go increment(&w, ch)}w.Wait()fmt.Println("final value of x", x)}// Final value of x 1000

技术小白记录学习过程,有错误或不解的地方请指出,如果这篇文章对你有所帮助请点点赞收藏+关注谢谢支持 !!!

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

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

相关文章

IO进程线程,文件与目录,实现linux任意目录下ls -la

注意文件的名字、路径是如何输入的。 函数opendir打开目录&#xff0c;struct dirent&#xff0c;struct stat这些结构体的含义。 readdir()函数是一个用于读取目录内容的系统调用或库函数&#xff0c;在类Unix操作系统中&#xff08;如Linux&#xff09;广泛使用。它用于遍历…

python爬虫10:selenium库

python爬虫10&#xff1a;selenium库 前言 ​ python实现网络爬虫非常简单&#xff0c;只需要掌握一定的基础知识和一定的库使用技巧即可。本系列目标旨在梳理相关知识点&#xff0c;方便以后复习。 申明 ​ 本系列所涉及的代码仅用于个人研究与讨论&#xff0c;并不会对网站产…

leetcode算法题--复杂链表的复制

原题链接&#xff1a;https://leetcode.cn/problems/fu-za-lian-biao-de-fu-zhi-lcof/description/?envTypestudy-plan-v2&envIdcoding-interviews 感觉一开始想到的办法还是比较笨 /*** Definition for a Node.* type Node struct {* Val int* Next *Node* …

C语言练习题Day1

从今天开始分享C语言的练习题&#xff0c;每天都分享&#xff0c;差不多持续16天&#xff0c;看完对C语言的理解可能更进一步&#xff0c;让我们开始今天的分享吧&#xff01; 题目一 执行下面的代码&#xff0c;输出结果是&#xff08;&#xff09; int x5,y7; void swap()…

Qt --- 对象树机制、析构顺序导致崩溃

{QWidget widget;QPushButton buttonQuit("Quit",&widget); } 作为父组件的widget&#xff0c;子组件buttonQuit都是QObject的子类&#xff1b;上述代码运行没有问题。 代码析构时&#xff0c;因为C的要求&#xff0c;局部对象的析构时按照实例化顺序逆向的顺序…

「MySQL-01」MySQL基础

目录 一、数据库概念 1. 什么是数据库 2. 为什么要有数据库&#xff1f; 3. 数据库将数据存在哪里&#xff1f; 二、知名数据库介绍 1.知名数据库介绍 2.为什么要学习MySQL 三、MySQL的基本使用 0. 安装MySQL 1. 数据库客户端链接服务端 2. Windows下的MySQL服务端管理 3. 数据…

昌硕科技、世硕电子同步上线法大大电子合同

近日&#xff0c;世界500强企业和硕联合旗下上海昌硕科技有限公司&#xff08;以下简称“昌硕科技”&#xff09;、世硕电子&#xff08;昆山&#xff09;有限公司&#xff08;以下简称“世硕电子”&#xff09;的电子签项目正式上线。上线仪式在上海浦东和硕集团科研大楼举行&…

flowable源码解析之分析环境搭建

系列文章目录 第一章 源码分析环境搭建 第二章 流程引擎启动(一) 第三章 流程引擎启动(二) 第四章 流程引擎启动(三) 第五章 流程引擎启动(四) 第六章 流程部署 第七章 流程模型解析 第八章 流程的发起 第九章 数据存储机制 文章目录 系列文章目录一、Flowable源码分析…

使用fastjson2的@JSONField注解解决日期格式记录

最近在做一个三方对接&#xff0c;对方的日期格式数据要求时间日期格式: yyyyMMddHHmmss或者 yyyyMMdd&#xff0c;我一下想起了fastjson2工具包&#xff0c;所以很愉快的解决了此问题。 依赖jar如下&#xff1a; <dependency><groupId>com.alibaba.fastjson2</…

mysql 触发器格式,修改触发器,创建触发器,触发条件

1.查询所有触发器 show triggers; 2.创建触发器 #修改结束符 delimiter $$create trigger <自定义出发名> after insert on <所监听的表名> for each row begin#执行多条sql语句#拿到监听数据 关键字 new/old 即 新数据/旧数据update classify set cat_num cat…

每日一博 - 闲聊云原生和容器编排

文章目录 概念1. 云原生&#xff08;Cloud Native&#xff09;&#xff1a;2. 容器编排&#xff08;Container Orchestration&#xff09;&#xff1a; 小结 概念 云原生和容器编排是两个不同的概念&#xff0c;但它们之间有着密切的联系。下面将分别介绍这两个概念&#xff0…

day-02 套接字类型与协议设置、地址族和数据序列

一.套接字类型 1.面向连接的套接字&#xff08;SOCK_STREAM&#xff09; 特点&#xff1a; 传输过程中数据不会消失按顺序传输数据传输的数据不存在数据边界 2.面向消息的套接字&#xff08;SOC_DGRAM&#xff09; 特点&#xff1a; 强调快速传输而非传输顺序传输的数据可…

【C++】const成员 | 取地址运算符重载

Ⅰ. const成员 两种const 我们知道&#xff0c;用const修饰 能起到保护&#xff0c;使之不被修改的作用。 修饰指针的const有两种位置&#xff1a; 我们学过的this指针&#xff0c;就被后者所修饰&#xff0c;因此无法被修改。 const成员函数 ➡️为了保护函数里的成员&…

《Java极简设计模式》第05章:原型模式(Prototype)

作者&#xff1a;冰河 星球&#xff1a;http://m6z.cn/6aeFbs 博客&#xff1a;https://binghe.gitcode.host 文章汇总&#xff1a;https://binghe.gitcode.host/md/all/all.html 源码地址&#xff1a;https://github.com/binghe001/java-simple-design-patterns/tree/master/j…

【日常积累】Linux下sftp搭建

概述 SFTP是Secure File Transfer Protocol的缩写&#xff0c;是安全文件传送协议。可以为传输文件提供一种安全的加密方法。跟ftp几乎语法功能一样。 SFTP是SSH的一部分&#xff0c;是一种传输档案至Blogger伺服器的安全方式。它本身没有单独的守护进程&#xff0c;必须使用s…

Elasticsearch 8.X reindex 源码剖析及提速指南

1、reindex 源码在线地址 为方便大家验证&#xff0c;这里给出 reindex github 源码地址。 https://github.com/elastic/elasticsearch/blob/001fcfb931454d760dbccff9f4d1b8d113f8708c/server/src/main/java/org/elasticsearch/index/reindex/ReindexRequest.java reindex 常见…

开源软件的崛起:历史与未来

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

Lambda函数

一.概念 1.利用lambda表达式可以编写内嵌的匿名函数&#xff0c;用以替换独立函数或者函数对象 2.每当你定义一个lambda表达式后&#xff0c;编译器会自动生成一个匿名类&#xff08;这个类当前重载了&#xff08;&#xff09;运算符&#xff09;&#xff0c;我们称为闭包类型…

8.6 【C语言】返回指针值的函数

一个函数可以返回一个整型值、字符值、实型值等&#xff0c;也可以返回指针型的数据&#xff0c;即地址。 类型名 *类型名&#xff08;参数表列&#xff09; 例8.25 有a个学生&#xff0c;每个学生有b门课程的成绩。要求在用户输入学生序号以后&#xff0c;能输出该学生的全部…

Lazada商品详情接口 获取Lazada商品详情数据 Lazada商品价格接

一、引言 随着电子商务的迅速发展和普及&#xff0c;电商平台之间的竞争也日趋激烈。为了提供更好的用户体验和更高效的后端管理&#xff0c;Lazada作为东南亚最大的电商平台之一&#xff0c;开发了一种商品详情接口&#xff08;Product Detail API&#xff09;。该接口允许第…