后端实践--go并发编程 青训营

简介: 后端实践--go并发编程 青训营

后端实践--go并发编程 青训营

go的擅长领域就是高并发编程!!由于go语言的出现时间较晚,已经是并发编程较为普及的时代,所以go 语言在设计的时候就天生支持并发。

先了解一下并发和并行的概念。并发是指在一个时间段内多个任务共同推进,如核心上跑多个任务,一般会由于时间片轮转调度这多个任务在一个时间段内看上去同时在推进;而并行是指安排多个核心,每个核心去执行者多个任务中的一个或多个。

go 通过goroutine 实现并发。goroutine类似于线程,属于==用户态的线程==,比内核态的线程更轻量。goroutinego 运行时调度完成的,而内核态线程是由OS调度的。

一般goroutine 配合函数使用。相当于启动一个新的线程去执行这个函数。

channel: 在多个goroutine通信

1. goroutine

func hello(){
  fmt.Println("aaaa")
}
// 这里会创建一个主goroutine
// 主goroutine退出了,那么其他从属的goroutine也就退出了
func main(){
  // 启动一个新的goroutine(线程)执行这个函数
  go hello()
  // 主执行流正常往下执行
  fmt.Println("bbb")
  tmie.Sleep(time.Second)
}

等待goroutine 结束 (有点像信号量)

// 声明全局等待组变量
var wg sync.WaitGroup
func hello() {
  defer wg.Done() // 告知当前goroutine完成   -- 操作
  fmt.Println("hello")  
}
func main() {
  wg.Add(1) // 登记1个goroutine             ++操作
  go hello()
  fmt.Println("你好")
  wg.Wait() // 阻塞等待登记的goroutine完成    等到wg减到0
}

goroutine调度 --- GMP调度系统

G: goroutine 每执行一次go f()就创建一个 G,包含要执行的函数和上下文信息。

M: Machine 实际干活的。和内核线程是一一映射的关系。

P 管理者,管理goroutine。管理 goroutine 执行所需的资源,最多有 GOMAXPROCS 个。

GOMAXPROCS

// 设置执行当前任务的CPU核心数,默认是所有的核心
runtime.GOMAXPROCS(4)
runtime.NumCPU()

goroutineOS线程的区别

  1. 一个os线程可以对应多个goroutine
  2. go程序也可以同时使用多个os 线程
  3. 他们是多对多的关系

2. channel

就可以理解为一个类型。--- 引用类型!

三个常用的引用类型:切片、mapchan ==需要make 初始化。==

实现多 goroutine间通信。

有点像管道通信,遵循先入先出原则。

声明通道但没有初始化时其为nil,这时读取和写入都会都会阻塞。

自带访问控制机制:==通道开启时==,当通道为空时取值会阻塞住;当通道为满时写入也会阻塞住。通道关闭且有缓冲区时:不能写入,但是可以读取值直到读完。

// 声明一个channel管道,只能存放int类型的变量
var a chan int
var b chan []string
...
// 初始化(不带缓冲区)
a = make(chan int)
// 初始化(带缓冲区),代缓冲区的通道可以把接收的的数据先存到通道里
b = make(chan []string, 16)

chan 操作 <-

向通道里写:a <- 1

从通道里读:ch := <- a

关闭通道:close(a) , 这个别忘了!

// 循环取值
a := make(chan int, 10)
// 一样的:
// 若通道是开启的没有close,那么这里取完值后会阻塞!!!
// 只有通道关闭时取完值后会返回退出循环
for i := range a{
  // ...
}
// 手动判断停止
for{
  // 若通道是开启的没有close,那么这里取完值后会阻塞!!!
  // 只有通道关闭时取完值后会返回:对应类型的零值和 false
  x,ok := <- a
  if !ok{
    // 读完了
    break
  }
  // ...
}

确保只执行一次。

var once sync.Once
// ...
// 确保只关闭一次,关闭已经关闭了的chan会引发panic!!
once.Do(func(){ close(a) })

单向通道

通常用于函数参数,限制为只允许读取或只允许写入的通道。

// 只能写入数据的通道
var a chan<- int
// 只能读取数据的通道
var b <-chan int

通道仅用来作为标记等时可以这样用,节省空间。

// 通道类型是空结构体,更节省空间!
var notice = make(chan struct{}, 10)
// ...
// 向notice通道里写入。  struct{} 是类型,再加一个 {} 表示实例化
notice <- struct{}{}

3. worker pool (goroutine 池)

就是开启多个 goroutine 竞争执行任务。

4. select 多路复用

同一时刻有多个通道,选择其中某一个执行。

让其他未满足条件的通道同时等待,提高效率。

a := make(chan int, 1)
select{
  // 哪个满足条件就执行哪一个,有多个条件满足时随机执行一个!
  case x := <- a: 
    // ...
  case a <- 10:
    // ... 
}

以上就是go并发编程的简单知识,另外涉及到并发,就还有互斥锁和读写锁等各类锁的使用以及线程互斥和线程同步的知识,这里就不赘述了。

相关文章
|
2月前
|
消息中间件 缓存 NoSQL
Redis各类数据结构详细介绍及其在Go语言Gin框架下实践应用
这只是利用Go语言和Gin框架与Redis交互最基础部分展示;根据具体业务需求可能需要更复杂查询、事务处理或订阅发布功能实现更多高级特性应用场景。
281 86
|
4月前
|
分布式计算 算法 安全
Go语言泛型-泛型约束与实践
Go语言中的泛型约束用于限制类型参数的范围,提升类型安全性。通过接口定义约束,可实现对数值类型、排序与比较等操作的支持。开发者既可使用标准库提供的预定义约束,如constraints.Ordered和constraints.Comparable,也可自定义约束以满足特定需求。泛型广泛应用于通用数据结构(如栈、队列)、算法实现(如排序、查找)及构建高效可复用的工具库,使代码更简洁灵活。
|
5月前
|
设计模式 人工智能 Go
go 依赖注入实践
依赖注入(DI)是一种软件设计模式,旨在降低代码耦合度,提高代码可测试性和可复用性。其核心思想是将依赖项从外部传入使用对象,而非由其内部创建。通过 DI,模块间关系更清晰,便于维护和扩展。常见实现包括方法注入和接口注入,适用于如 Go 等支持函数式编程和接口抽象的语言。
132 8
|
5月前
|
开发框架 JSON 中间件
Go语言Web开发框架实践:路由、中间件、参数校验
Gin框架以其极简风格、强大路由管理、灵活中间件机制及参数绑定校验系统著称。本文详解其核心功能:1) 路由管理,支持分组与路径参数;2) 中间件机制,实现全局与局部控制;3) 参数绑定,涵盖多种来源;4) 结构体绑定与字段校验,确保数据合法性;5) 自定义校验器扩展功能;6) 统一错误处理提升用户体验。Gin以清晰模块化、流程可控及自动化校验等优势,成为开发者的优选工具。
|
6月前
|
存储 消息中间件 前端开发
PHP后端与uni-app前端协同的校园圈子系统:校园社交场景的跨端开发实践
校园圈子系统校园论坛小程序采用uni-app前端框架,支持多端运行,结合PHP后端(如ThinkPHP/Laravel),实现用户认证、社交关系管理、动态发布与实时聊天功能。前端通过组件化开发和uni.request与后端交互,后端提供RESTful API处理业务逻辑并存储数据于MySQL。同时引入Redis缓存热点数据,RabbitMQ处理异步任务,优化系统性能。核心功能包括JWT身份验证、好友系统、WebSocket实时聊天及活动管理,确保高效稳定的用户体验。
425 4
PHP后端与uni-app前端协同的校园圈子系统:校园社交场景的跨端开发实践
|
5月前
|
开发框架 安全 前端开发
Go Web开发框架实践:模板渲染与静态资源服务
Gin 是一个功能强大的 Go Web 框架,不仅适用于构建 API 服务,还支持 HTML 模板渲染和静态资源托管。它可以帮助开发者快速搭建中小型网站,并提供灵活的模板语法、自定义函数、静态文件映射等功能,同时兼容 Go 的 html/template 引擎,具备高效且安全的页面渲染能力。
|
5月前
|
开发框架 JSON 中间件
Go语言Web开发框架实践:使用 Gin 快速构建 Web 服务
Gin 是一个高效、轻量级的 Go 语言 Web 框架,支持中间件机制,非常适合开发 RESTful API。本文从安装到进阶技巧全面解析 Gin 的使用:快速入门示例(Hello Gin)、定义 RESTful 用户服务(增删改查接口实现),以及推荐实践如参数校验、中间件和路由分组等。通过对比标准库 `net/http`,Gin 提供更简洁灵活的开发体验。此外,还推荐了 GORM、Viper、Zap 等配合使用的工具库,助力高效开发。
|
5月前
|
数据采集 安全 Go
Go 语言并发编程基础:Goroutine 的创建与调度
Go 语言的 Goroutine 是轻量级线程,由 runtime 管理,具有启动快、占用小、支持高并发的特点。本章介绍 Goroutine 的基本概念、创建方式(如使用 `go` 关键字或匿名函数)、M:N 调度模型及其工作流程,并探讨其在高并发场景中的应用,帮助理解其高效并发的优势。
|
6月前
|
设计模式 缓存 算法
Go如何进行高质量编程与性能调优实践
本文介绍了Go语言高质量编程与性能调优的实践方法。高质量编程包括良好的编码习惯(如清晰注释、命名规范)、代码风格与设计(如MVC模式)、简洁明了的代码原则,以及单元测试与代码重构的重要性。性能调优方面,涵盖算法优化、数据结构选择、I/O优化、内存管理、并行与并发处理优化及代码层面的改进。通过这些方法,可有效提升代码质量和系统性能。
151 13
|
5月前
|
Go 开发者
Go 并发编程基础:无缓冲与有缓冲通道
本章深入探讨Go语言中通道(Channel)的两种类型:无缓冲通道与有缓冲通道。无缓冲通道要求发送和接收必须同步配对,适用于精确同步和信号通知;有缓冲通道通过内部队列实现异步通信,适合高吞吐量和生产者-消费者模型。文章通过示例对比两者的行为差异,并分析死锁风险及使用原则,帮助开发者根据场景选择合适的通道类型以实现高效并发编程。

热门文章

最新文章