go语言并发编程(中)

news2025/1/13 17:29:54

go语言并发编程中

  • 并发安全和锁
    • 互斥锁Mutex
    • 读写锁
    • 信号量&条件变量
    • 单例模式&time包方法
  • 协程池

并发安全和锁

互斥锁Mutex

有时候在Go代码中可能会存在多个goroutine同时操作一个资源(临界区),这种情况会发生竞态问题(数据竞态)。类比现实生活中的例子有十字路口被各个方向的的汽车竞争;还有火车上的卫生间被车厢里的人竞争。此时我们需要保证这个线程安全的问题的话我们需要使用这个互斥锁来保证这个线程安全的问题,锁再其他语言当中都是有的再这里就不过多解释了。下面举一个列子来演示一下这个go语言当中sync包里面的互斥锁。

package main

import (
	"fmt"
	"sync"
)

var x int64 //系统默认会初始化为对应的零值
var wg sync.WaitGroup
func Add() {
	
	for i := 0; i < 200; i++ {
		x = x + 1
	}
	wg.Done()
}

func main() {

	wg.Add(3)

	go Add()
	go Add()
	go Add()
	wg.Wait()
	fmt.Println(x)

}

此时这个x是这个临界资源多个执行流都是能够看到的,如果我们不加锁进行保护很有可能造成结果和我们预期的不一样所以我们需要加锁进行保护。

package main

import (
	"fmt"
	"sync"
)

var x int64 //系统默认会初始化为对应的零值
var wg sync.WaitGroup

var mtx sync.Mutex

func Add() {
	defer wg.Done()
	mtx.Lock()
	for i := 0; i < 200; i++ {
		x = x + 1
	}
	mtx.Unlock()

}

func main() {
	wg.Add(3)
	go Add()
	go Add()
	go Add()
	wg.Wait()
	fmt.Println(x)
}

读写锁

再实际情况当中我们也有可能出现这个读的多但是这个线的少此时如果我们使用这个互斥锁的话效率非常的低下。因此在读多写少的环境中,可以优先使用读写互斥锁(sync.RWMutex),它比互斥锁更加高效。sync 包中的 RWMutex 提供了读写互斥锁的封装。下面我们简单介绍一下这个读写锁。

读写锁分为两种:读锁和写锁。当一个goroutine获取读锁之后,其他的goroutine如果是获取读锁会继续获得锁,如果是获取写锁就会等待;当一个goroutine获取写锁之后,其他的goroutine无论是获取读锁还是写锁都会等待。
其主要特点是这个读读不互斥,这个读写互斥。下面我们通过一个案例演示一下这个读写锁的使用。再使用之前解释一下go语言当中RWMutex的几个方法

     Lock()//加写锁
     UnLock()//解锁
     RLock()//加读锁
     RUnlock()//解锁

下面这段代码是用来验证这个两边都加的是这个写锁验证这个是互斥的

package main

import (
	"fmt"
	"sync"
	"time"
)

func main() {

	var locker sync.RWMutex
	var wg sync.WaitGroup
	wg.Add(2)

	go func() {
		defer wg.Done()
		fmt.Println("A lock before")
		locker.Lock()
		fmt.Println("A locked")
		time.Sleep(time.Second * 2)
		fmt.Println("A:Unlock")
		locker.Unlock()

	}()

	//使用读写锁
	go func() {
		defer wg.Done()
		fmt.Println("B lock before")
		locker.Lock()
		fmt.Println("B locked")
		time.Sleep(time.Second * 2)
		fmt.Println("B:Unlock")
		locker.Unlock()

	}()

	wg.Wait()
}

在这里插入图片描述
我们发现这个写写是互斥的。下面我们将其改为这个读写进一步验证我们的猜想

package main

import (
	"fmt"
	"sync"
	"time"
)

func main() {

	var locker sync.RWMutex
	var wg sync.WaitGroup
	wg.Add(2)

	go func() {
		defer wg.Done()
		fmt.Println("A lock before")
		locker.Lock()
		fmt.Println("A locked")
		time.Sleep(time.Second * 2)
		fmt.Println("A:Unlock")
		locker.Unlock()

	}()

	//使用读写锁
	go func() {
		defer wg.Done()
		fmt.Println("B lock before")
		locker.RLock()
		fmt.Println("B locked")
		time.Sleep(time.Second * 2)
		fmt.Println("B:Unlock")
		locker.RUnlock()

	}()
	wg.Wait()
}

在这里插入图片描述
通过运行结果我们可以看出来这个我们发现这个读写也是互斥的。

下面我们验证的是这个如果A和B如果同时加的是这个读锁是否会互斥了?

package main

import (
	"fmt"
	"sync"
	"time"
)

func main() {

	var locker sync.RWMutex
	var wg sync.WaitGroup
	wg.Add(2)

	go func() {
		defer wg.Done()
		fmt.Println("A lock before")
		locker.RLock()
		fmt.Println("A locked")
		time.Sleep(time.Second * 2)
		fmt.Println("A:Unlock")
		locker.RUnlock()

	}()

	//使用读写锁
	go func() {
		defer wg.Done()
		fmt.Println("B lock before")
		locker.RLock()
		fmt.Println("B locked")
		time.Sleep(time.Second * 2)
		fmt.Println("B:Unlock")
		locker.RUnlock()

	}()

	wg.Wait()
}

对应运行结果
在这里插入图片描述
我们发现读读是不互斥的结果和我们上面的猜想是一样的非常的一致。上述我也只演示了这个读写锁的使用。再这里再次强调一下这个需要注意的是读写锁非常适合读多写少的场景,如果读和写的操作差别不大,读写锁的优势就发挥不出来。

信号量&条件变量

再刚刚的代码当中我们提前使用了这个条件变量。再这里我们介绍一下这个信号量的使用
在代码中生硬的使用time.Sleep肯定是不合适的,Go语言中可以使用sync.WaitGroup来实现并发任务的同步。 sync.WaitGroup有以下几个方法:

 (wg * WaitGroup) Add(delta int) 
 (wg *WaitGroup) Done()
 (wg *WaitGroup) Wait()

下面解释一下这三个函数的意思是什么?下面一个一个的来解释一下其含义

  • 计数器+delta
  • 计数器-1
  • 阻塞直到计数器变为0

注意这个信号量的本质是一把这个计数器sync.WaitGroup内部维护着一个计数器,计数器的值可以增加和减少。例如当我们启动了N 个并发任务时,就将计数器值增加N。每个任务完成时通过调用Done()方法将计数器减1。通过调用Wait()来等待并发任务执行完,当计数器值为0时,表示所有并发任务已经完成。下面我们再简单的演示一下这个使用吧上面已经使用过几次了非常的简单如果使用过其他的这个信号量的话

var wg sync.WaitGroup

func hello() {
	defer wg.Done()
	fmt.Println("Hello Goroutine!")
}
func main() {
	wg.Add(1)
	go hello() // 启动另外一个goroutine去执行hello函数
	fmt.Println("main goroutine done!")
	wg.Wait()
}

下面我们来看看这个条件变量的使用go语言当中的Cond。其实和其他语言当中的差不多。下面我们通过一段代码看看这个go语言当中的条件变量如何使用

package main

import (
	"fmt"
	"sync"
	"time"
)

func main() {
	var locker sync.Mutex
	var wg sync.WaitGroup
	wg.Add(2)
	cond := sync.NewCond(&locker)
	go func() {
		cond.L.Lock() //注意需要提前获取锁再wait之前
		defer cond.L.Unlock()
		fmt.Println("Conde wait")
		cond.Wait()
		fmt.Println("Cond After")
		wg.Done()

	}()

	go func() {
		defer wg.Done()
		time.Sleep(time.Second * 3)
		fmt.Println("notice")
		//cond.Broadcast() //唤醒所有再条件变量下等待的
		cond.Signal() //唤醒一个
	}()

	wg.Wait()

	fmt.Println("main over")

}

注意再使用条件变量的时候我们必须要先获取锁之后采用使用这个wait。下面我们来看看这个运行结果
在这里插入图片描述

单例模式&time包方法

这个单例模式再博主以前的博客当中已经写过了。再这里我们看看go语言当中的单例模式是如何进行书写的。再go语言当中利用这个sync包里面的Once

package main

import "sync"

type Singleton struct {
	Name string
	age  int
}

var once sync.Once
var inst *Singleton

func GetInstance() *Singleton {
	once.Do(func() {
		inst = &Singleton{}
	})
	return inst
}
func main() {

}

在编程的很多场景下我们需要确保某些操作在高并发的场景下只执行一次,例如只加载一次配置文件、只关闭一次通道等。Go语言中的sync包中提供了一个针对只执行一次场景的解决方案–sync.Once。sync.Once只有一个Do方法,其签名如下:

func (o *Once) Do(f func()) {}

正好这个单例模式就是这样的。其他经典的写法由于这个博主再单例模式的那篇博客当中已经写了再这里就不重复了。
其实这个sync.Once里面的实现其实也是这个双重加锁。下面我们来看看这个他的实现吧

func (o *Once) Do(f func()) {   //判断是否执行过该方法,如果执行过则不执行
    if atomic.LoadUint32(&o.done) == 1 {
        return
    }
    // Slow-path.
    o.m.Lock()
    defer o.m.Unlock()  //进行加锁,再做一次判断,如果没有执行,则进行标志已经扫行并调用该方法
    if o.done == 0 {
        defer atomic.StoreUint32(&o.done, 1)
        f()
    }
}

下面我们来学习一下这个go语言当中这个 time包里面的两个方法一个是这个After一个是这个Tick

在等待给定的一段时间后,向返回值发送当前时间,返回值是一个单向只读通道.下面我们通过一个小小的demo来展示一下他的功能

func main() {
	c := make(chan int, 1)

	select {
	case m := <-c:
		fmt.Println(m)
	case d := <-time.After(5 * time.Second):
		fmt.Println("time out")
		fmt.Println("current Time :", d)
	}
	fmt.Println("main over")

}

运行发现,在等待五秒钟后,打印

在这里插入图片描述
第一个case中,通道c一直处于阻塞状态;第二个case中,在time.After()结束后,从返回值通道中取出一个值赋予了d,该值就是当前时间。
下面我们自己模拟实现一下这个time包里面的After是怎么实现的。其实非常的简单

//time.After的实现
func timeAfter(interval time.Duration) <-chan time.Time {
	timeChannel := make(chan time.Time)
	go func() {
		time.Sleep(interval)
		timeChannel <- time.Now()
	}()
	return timeChannel
}

下面我们再来看看这个time包里面的这个Tick

func tick(d Duration) <-chan Time

tick() 函数接受一个时长 d,然后它 tick() 等待 d 时长,等待时间到后,将等待完成时所处时间点写入到 channel 中并返回这个只读 channel。

package main
import (
	"fmt"
	"time"
)
func main() {
	fmt.Println("begin")
	select {
	case <-time.Tick(2 * time.Second):
		fmt.Println("2 second over:", time.Now().Second())
	case <-time.After(7 * time.Second):
		fmt.Println("5 second over, timeover", time.Now().Second())
		return
	}
}

上面的示例,在等待 2 秒之后,就会因为读取到了 time.Tick() 的通道数据而终止,因为 select 并未在循环内.
下面我们来看看他的这个简单的实现这个Tick

func timeTick(interval time.Duration) <-chan time.Time {
	timeChannel := make(chan time.Time)
	go func() {
		for {
			time.Sleep(interval)
			timeChannel <- time.Now()
		}
	}()
	return timeChannel
}

其实都非常的简单看起来很花里呼哨的,其实这个底层还是很简单的。下面我们探讨一下这个go语言当中的这个写循环执行的定时任务,常见的有以下三种实现方式:

首先我们看看第一种实现这个使用time包里面的sleep

for {

   time.Sleep(time.Second)

   fmt.Println("我在定时执行任务")

}

第二种方法time.Tick函数

t1:=time.Tick(3*time.Second)

for {

   select {

   case <-t1:

      fmt.Println("t1定时器")

   }

}

第三种方法是这个

t:=time.NewTicker(time.Second)

for {

   select {

   case <-t.C:

      fmt.Println("t1定时器")

      t.Stop()

   }

}

其中Tick定时任务,也可以先使用time.Ticker函数获取Ticker结构体,然后进行阻塞监听信息,这种方式可以手动选择停止定时任务,在停止任务时,减少对内存的浪费。
现在我们知道了,Tick,Sleep,包括time.After函数,都使用的timer结构体,都会被放在同一个协程中统一处理,这样看起来使用Tick,Sleep并没有什么区别。

实际上是有区别的,Sleep是使用睡眠完成定时任务,需要被调度唤醒。Tick函数是使用channel阻塞当前协程,完成定时任务的执行。当前并不清楚golang 阻塞和睡眠对资源的消耗会有什么区别,这方面不能给出建议。
但是使用channel阻塞协程完成定时任务比较灵活,可以结合select设置超时时间以及默认执行方法,而且可以设置timer的主动关闭,以及不需要每次都生成一个timer(这方面节省系统内存,垃圾收回也需要时间)。
所以个人建议使用time.Tick完成定时任务。

协程池

本文使⽤Go语⾔实现并发的协程调度池阉割版,本⽂主要介绍协程池的基本设计思路,⽬的为深⼊浅出快速了解协程池⼯作原理,与真实的企业协程池还有很⼤差距,本⽂仅供学习参考。

一.首先我们需要知道何谓并发?go语言是怎么实现并发?
在这里插入图片描述
在这里插入图片描述
1.
同⼀时刻可以处理多个事务
2.
更加节省时间,效率更⾼

下面我们来看看这个协程池的设计思路
为什么需要协程池?虽然go语⾔在调度Goroutine已经优化的⾮常完成,
并且Goroutine作为轻量级执⾏流程,也不需要CPU调度器的切换,我们⼀般在使⽤的时候,如果想处理⼀个分⽀流程,直接go⼀下即可。但是,
如果⽆休⽌的开辟Goroutine依然会出现⾼频率的调度Groutine,那么依然会浪费很多上下⽂切换的资源,导致做⽆⽤功。所以设计⼀个Goroutine
池限制Goroutine的开辟个数在⼤型并发场景还是必要的。
在这里插入图片描述
和这个线程池非常的类似再这里就不过多演示
对应代码实现

package main

import (
	"fmt"
	"time"
)

/*
   设计一个协程池
   Work和Task通过Channel完成通信
*/

type Task struct {
	Run func() error
	//回调方法
}

func NewTask(f func() error) *Task {
	t := Task{Run: f}
	return &t
}

func (t *Task) Execute() {
	t.Run()
}

/*
   上面是有关任务的方法
*/

type Pool struct {
	//对外Task入口
	EntryChannel chan *Task
	//管道当中放入这个任务
	//内部Task队列,JobChannel
	JobsChannel chan *Task
	workNum     int
}

func (p *Pool) Work(WorkId int) {
	//一直往Job管道当中读取任务,并执行任务

	for {
		task, ok := <-p.JobsChannel
		if ok {
			task.Execute()
			fmt.Printf("协程%d执行让任务 \n", WorkId)
			time.Sleep(time.Second)
		}
	}

}
func (p *Pool) run() {

	//根据创建协程池的数目来创建协程
	//从EntryChannel当中取任务将取到的任务发送给JobsChannel

	for i := 0; i < p.workNum; i++ {
		go p.Work(i)
	}

	for {
		task, ok := <-p.EntryChannel
		if ok {
			p.JobsChannel <- task
		}
	}
}

// NewPool 创建协程池
func NewPool(num int) *Pool {
	p := Pool{
		EntryChannel: make(chan *Task),
		JobsChannel:  make(chan *Task),
		workNum:      num,
	}
	return &p
}

func main1() {

	t := NewTask(func() error {
		fmt.Println(time.Now())

		return nil
	})

	p := NewPool(4)

	go func() {
		for {
			p.EntryChannel <- t
		}
	}()
	//注意这个需要写到后面
	p.run()
}

/*
   进程和进程之间是独立的好处一个进程死亡消耗内存
   线程和线程之间的内存资源大部分是共享的。好处节省资源稳健性不强
   协程实际上是更轻量级别的线程
*/

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

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

相关文章

夹抱式伸缩货叉料箱多层穿梭车立体库|海格里斯HEGERLS高速穿梭车按需定制

近年来&#xff0c;随着互联网的不断普及&#xff0c;电子商务迅速发展起来&#xff0c;客户订单商品呈现出品种多、批量少、交货期短的特点&#xff0c;使得供货商物流配送中心货架存储密集度不断提升&#xff0c;物料出入库频率越来越高&#xff0c;小件拆零拣选量与日俱增&a…

m基于5G毫米波场景Salen-Valenzula信道建模与matlab仿真

目录 1.算法描述 2.仿真效果预览 3.MATLAB核心程序 4.完整MATLAB 1.算法描述 随着通信技术的不断发展&#xff0c;现有的通信系统已经无法满足人们各个方面的需求&#xff0c;为了解决这个问题&#xff0c;5G通信技术应运而生。5G技术拥有更高的通信传输速率&#xff0c;更…

如何实现一套优雅的Baas查询语言?

Baas平台内&#xff0c;同一区块链底层接口数据可能会被应用在不同的业务场景下&#xff0c;因此需要进行筛选或者排序等操作。为满足此类需求&#xff0c;通常后端需要开发新的接口或增加新的字段以满足业务诉求。但随着需求业务不断变更&#xff0c;往往不可避免地导致接口的…

Activiti初识

以前 在没有工作流引擎的时候&#xff0c;要实现流程控制&#xff0c;我们需要在数据库中定义表&#xff0c;然后采用状态字段去跟踪流程的变化&#xff1a;比如是否到下一个流程&#xff1b; 然后到下一个角色执行的时候&#xff0c;我们需要判断用户是否具有审批的权限&…

HFSS使用经验

目录 一、如何找到Project工程变量 二、coverlines是什么东西 三、参数建模的第一步 四、让HFSS模型居中的快捷键CtrlD 五、关于介质颜色的修改 六、如何在HFSS中添加变量 七、如何绘制微带线 八、如何绘制带缺口的微带线谐振腔 九、如何设置激励源 十、HFSS如何镜像…

关于一致性问题的简单总结

一、场景引入 在随着人类的发展过程&#xff0c;由于人类是一群人&#xff0c;而非单个人&#xff0c;并相互联系&#xff0c;为了基本的生存或更美好的精神追求&#xff0c;出现了分工协作&#xff08;单个人无法完成&#xff09;的概念。人性是具有追求极致、美化的特性&…

【SQL】字符串处理函数

文章目录修复表中的名字left(string, count)substr(string, pos, len)按日期分组销售产品group_concat患某种疾病的患者修复表中的名字 将name的首字母大写&#xff0c;其余字母小写 select user_id, concat(upper(left(name, 1)), lower(substr(name, 2))) as name from User…

【Python机器学习】神经网络中全连接层与线性回归的讲解及实战(Tensorflow、MindSpore平台 附源码)

需要全部代码请点赞关注收藏后评论区留言私信~~~ 全连接层与线性回归 神经网络模型也是参数学习模型&#xff0c;因为对它的学习只是得到神经网络参数的最优值&#xff0c;而神经网络的结构必须事先设计好。如果确实不能通过改进学习过程来达到理想效果&#xff0c;则要重新设…

linux基础学习-用户权限相关命令

用户权限相关命令 用户和权限的基本概念 基本概念 用户是linux系统工作中的重要的一环&#xff0c;用户管理包括 用户 和 组 管理在linux系统下&#xff0c;不论是由本机还是远程登录系统&#xff0c;每个系统都必须有一个账号&#xff0c;并且对于不同的系统资源拥有不同的使用…

银河系中心黑洞的首张照片

说到黑洞&#xff0c;那就不得不提起我们的家园银河系中心的大黑洞&#xff0c;在今天这张照片出来之前&#xff0c;所有关于银河系黑洞的描述都是推测、理论&#xff0c;而今天成为了现实&#xff01; 2019年&#xff0c;同一团队拍摄了梅西耶87星系&#xff08;M87&#xff0…

文本分类优化方法

文本分类优化方法 文本分类是NLP的基础工作之一&#xff0c;也是文本机器学习中最常见的监督学习任务之一&#xff0c;情感分类&#xff0c;新闻分类&#xff0c;相似度判断、问答匹配、意图识别、推断等等领域都使用到了文本分类的相关知识或技术。文本分类技术在机器学习的发…

用 Markdown 快速生成漂亮的 Latex 伪代码

参考&#xff1a;在 Markdown 中书写伪代码 文章目录配置 VSCode编写 Latex 源码生成 pseudocode配置 VSCode 组合&#xff1a; VSCode Markdown Preview Enhanced pseudocode.js 安装好 VSCode 和 Markdown Preview Enhanced 插件 按下快捷键 Ctrl Shift P&#xff0c;打…

定时器/计数器的基本概念

80C51单片机中有两个计数器&#xff0c;即T0和T1。 单片机内有一个定时器/计数器T0&#xff0c;可以用编程的方法将它设为计数器。当用作计数器时&#xff0c;它是一个16位计数器&#xff0c;它的最大计数值为65536。 定时器/计数器T0和T1分别是由TH0、TL0和TH1、TL1两个8位计数…

Vue3与Vue2生命周期不同点

一、前言 随着Vue3发布了两年多的时间&#xff0c;越来越多的小伙伴已经将老项目中的Vue2版本进行升级或者在新项目中使用到了Vue3.x的版本&#xff0c;今天就来总结以下Vue3相较于Vue2升级的生命周期不同点在哪。 二、生命周期 下面是生命周期对比图&#xff1a; Vue2Vue3…

JMeter基础入门

目录&#xff1a;导读 一、概述 二、Jmeter目录文件讲解 结语 一、概述 JMeter是Apache下一款在国外非常流行和受欢迎的开源性能测试工具&#xff0c;JMeter可用于模拟大量负载来测试一台服务器&#xff0c;网络或者对象的健壮性或者分析不同负载下的整体性能。 1、压测不同…

ValidateCode验证码的使用详解(初学看完都会用)

✅作者简介&#xff1a;热爱国学的Java后端开发者&#xff0c;修心和技术同步精进。 &#x1f34e;个人主页&#xff1a;Java Fans的博客 &#x1f34a;个人信条&#xff1a;不迁怒&#xff0c;不贰过。小知识&#xff0c;大智慧。 &#x1f49e;当前专栏&#xff1a;Java案例分…

Android 进阶——性能优化之电量优化全攻略及实战小结(二)

文章大纲引言一、在低电耗模式和应用待机模式下进行测试1、在低电耗模式下测试您的应用2、在应用待机模式下测试您的应用3、列入白名单的可接受用例4、确定当前充电状态5、监控充电状态变化6、确定当前电池电量7、监控显著的电池电量变化二、Wakelock 机制1、WakeLock分类2、申…

Linux系统x86-64架构下,从零实现一个系统调用。Ubuntu22.04LTS

名称版本OSUbuntu 22.04 LTSCurrent Kernel5.15.0-56-genericDestination Kernel5.16.60首先要会编译linux内核的源码,这块在我的另外一片文章里面。 https://blog.csdn.net/jl19861101/article/details/128327069 打开linux内核源码目录/arch/x86/entry/syscalls/syscall_64.t…

前端面试比较好的回答

介绍一下Connection:keep-alive 什么是keep-alive 我们知道HTTP协议采用“请求-应答”模式&#xff0c;当使用普通模式&#xff0c;即非KeepAlive模式时&#xff0c;每个请求/应答客户和服务器都要新建一个连接&#xff0c;完成 之后立即断开连接&#xff08;HTTP协议为无连接…

【笔记】canvas 绘制足球 —— 第一步 画个球体

文章目录一、球体分析二、足球结构分析三、canvas常用API四、画个球体1.初始化2.代码五、加上足球的皮肤一、球体分析 先上两张图 球坐标转直角坐标 xρsin(φ)cos(θ)x \rho \times sin(\varphi) \times cos(\theta) xρsin(φ)cos(θ) yρsin(φ)sin(θ)y \rho \times si…