go并发模型的详细介绍
Go 语言的并发模型是其一大亮点,它使得并发编程变得简单高效。Go 语言并发模型的核心概念是 goroutines 和 channels。在理解这两个概念之前,我们首先了解并发编程的一些基本概念。
1. 并发与并行
-
并发(Concurrency):并发是指系统能够在同一时间段内处理多个任务。并发并不意味着多个任务同时执行,而是指在一个时间段内,任务的切换和处理可以交替进行。并发通常是通过时间分片实现的(比如操作系统的时间片轮转)。
-
并行(Parallelism):并行是指多个任务同时在不同的处理器或核上执行。并行要求硬件支持多核心或者多处理器。
Go 语言的并发模型关注的是 并发,通过调度和轻量级的线程(即 goroutine)实现高效的并发执行。
2. Goroutines
Goroutine 是 Go 中并发执行的基本单元。它是一个由 Go 运行时管理的轻量级线程。相较于操作系统线程,goroutine 的开销非常小,启动一个 goroutine 只需要几个 KB 的内存,因此可以同时创建成千上万个 goroutine。
启动 Goroutine
使用 go
关键字启动一个 goroutine。goroutine 会在后台并发执行一个函数。
package mainimport ("fmt""time"
)func sayHello() {fmt.Println("Hello from Goroutine")
}func main() {go sayHello() // 启动一个 goroutinetime.Sleep(1 * time.Second) // 等待 goroutine 执行完毕fmt.Println("Hello from main")
}
在这个例子中,go sayHello()
启动了一个新的 goroutine。由于主函数没有等待 goroutine 执行完,因此我们加了 time.Sleep
来确保主函数能等待 goroutine 执行完成。
Goroutine 特点:
- 轻量级:启动一个 goroutine 的开销很小。
- 由 Go 运行时管理:Go 运行时会自动调度 goroutine,可以在一个线程上调度数以千计的 goroutine。
- 栈动态增长:每个 goroutine 的栈大小非常小,通常是 2 KB。随着需要,栈会自动增长。
- 非阻塞:如果 goroutine 执行的代码块遇到阻塞,它会被调度到其他的 goroutine,直到阻塞解决。
Goroutine 例子
package mainimport "fmt"func printMessage(msg string) {fmt.Println(msg)
}func main() {go printMessage("Hello from Goroutine 1")go printMessage("Hello from Goroutine 2")fmt.Println("Hello from main")
}
这里启动了两个 goroutine,它们与主函数并发执行,输出的顺序不固定,取决于调度器的执行。
3. Channel
Go 语言的并发模型不仅仅依赖于 goroutines,还依赖于 channels。channels 是一种数据结构,它允许 goroutine 之间进行通信和同步。
Channel 的基本使用
可以通过 make(chan Type)
创建一个 channel。Go 中的 channel 是类型安全的,即它只能传递指定类型的数据。
package mainimport "fmt"func greet(ch chan string) {ch <- "Hello from Goroutine"
}func main() {ch := make(chan string) // 创建一个 channelgo greet(ch) // 启动一个 goroutine,传递数据到 channelmsg := <-ch // 从 channel 中接收数据fmt.Println(msg)
}
在这个例子中:
ch <- "Hello from Goroutine"
:将数据发送到 channel。msg := <-ch
:从 channel 中接收数据。
Channel 的特性:
- 同步:channel 的发送和接收操作是阻塞的,直到数据被发送或接收。
- 类型安全:channel 必须指定类型,不能跨类型传递数据。
- 双向和单向:channel 可以作为双向通信的通道,也可以指定为只发送(
chan<-
)或只接收(<-chan
)。
Buffered Channel(缓冲区)
Go 的 channel 不一定需要是无缓冲的。你可以创建带缓冲区的 channel,这样可以在 channel 没有接收者的情况下继续发送数据,直到缓冲区满。
ch := make(chan string, 2) // 创建一个缓冲区大小为 2 的 channel
ch <- "Hello"
ch <- "Go"
fmt.Println(<-ch)
fmt.Println(<-ch)
带缓冲区的 channel 不会立即阻塞发送方,直到缓冲区满才会阻塞。
4. Goroutines 与 Channels 协作
Goroutines 和 Channels 配合使用时,可以实现非常高效的并发程序设计。通过 channels,goroutines 可以安全地交换数据和同步执行。
示例:生产者-消费者模式
生产者消费者模式是并发编程中常见的设计模式,其中生产者负责生成数据,消费者负责处理数据。
package mainimport "fmt"func producer(ch chan int) {for i := 1; i <= 5; i++ {ch <- i // 将数据发送到 channel}close(ch) // 关闭 channel,表示生产者不再发送数据
}func consumer(ch chan int) {for num := range ch { // 从 channel 接收数据,直到 channel 被关闭fmt.Println("Consumed:", num)}
}func main() {ch := make(chan int)go producer(ch) // 启动生产者go consumer(ch) // 启动消费者fmt.Println("Main goroutine")
}
在这个例子中:
- 生产者 goroutine 通过
ch <- i
向 channel 发送数据。 - 消费者 goroutine 通过
num := range ch
从 channel 中接收数据。由于 channel 被关闭,消费者会停止接收。
5. Select 语句
select
是 Go 中用于多路复用的语句,可以同时等待多个 channel 操作的完成。它类似于传统编程语言中的 select
或 poll
。
示例:使用 select
处理多个 channel
package mainimport "fmt"func main() {ch1 := make(chan string)ch2 := make(chan string)go func() { ch1 <- "Hello from ch1" }()go func() { ch2 <- "Hello from ch2" }()select {case msg1 := <-ch1:fmt.Println("Received:", msg1)case msg2 := <-ch2:fmt.Println("Received:", msg2)}
}
在这个例子中,select
会阻塞并等待任一 channel 中的数据。如果 ch1
或 ch2
有数据,它会选择接收并执行相应的代码块。
6. 并发模式
Go 的并发模型支持多种并发模式,常见的并发模式包括:
- 工作池模式(Worker Pool):将任务分配给多个 goroutine 进行处理,任务和工作者之间通过 channel 进行通信。
- 生产者消费者模式(Producer-Consumer):生产者和消费者通过 channel 进行数据交换。
- Fan-out, Fan-in:多个 goroutine 从同一个 channel 中读取数据(fan-out),或者多个 goroutine 向同一个 channel 发送数据(fan-in)。
7. 并发安全
Go 提供了多种方式来保证并发程序的安全性:
- Mutex(互斥锁):可以通过
sync.Mutex
来保护共享资源。 - Atomic 操作:
sync/atomic
包提供了原子操作来保证并发安全。 - Channel:Channel 本身是并发安全的,它可以在 goroutine 之间安全地传递数据。
总结
Go 的并发模型是基于 goroutines 和 channels 构建的,具备高效、简单、易用的特点。通过 goroutine,Go 可以轻松实现并发执行,而通过 channel,它可以安全地在 goroutines 之间传递数据和进行同步。Go 的并发编程模型大大简化了并发程序的开发,减少了传统并发编程中常见的错误和复杂性,使得开发者可以集中精力解决实际业务问题。
相关文章:
go并发模型的详细介绍
Go 语言的并发模型是其一大亮点,它使得并发编程变得简单高效。Go 语言并发模型的核心概念是 goroutines 和 channels。在理解这两个概念之前,我们首先了解并发编程的一些基本概念。 1. 并发与并行 并发(Concurrency):…...
使用FreeNAS软件部署ISCSI的SAN架构存储(IP-SAN)练习题
一,实验用到工具分别为: VMware虚拟机,安装教程:VMware Workstation Pro 17 安装图文教程 FreeNAS系统,安装教程:FreeNAS-11.2-U4.1安装教程2024(图文教程) 二,新建虚…...
FreeSWITCH实现多人电话会议功能
FreeSWITCH实现多人电话会议功能 作者:基于Java与FreeSWITCH的开源呼叫中心系统FreeIPCC FreeSWITCH作为一个开源的电话软交换平台,为企业和运营商提供了构建高效、灵活的语音通信系统的能力。其中,多人电话会议功能是其核心应用之一&#…...
Chromium 中chrome.webRequest扩展接口定义c++
一、chrome.webRequest 注意:从 Manifest V3 开始,"webRequestBlocking"权限不再适用于大多数扩展程序。以"declarativeNetRequest"为例,它允许使用declarativeNetRequest API。除了"webRequestBlocking"之外…...
极乐 15.2.6 | 清爽版简约美观音乐软件,支持网易云歌单导入
极乐是一款使用起来非常轻松的音乐播放软件,它拥有清新简洁的画面,专注于音乐播放功能。最新版本全面升级了64位架构,带来了前所未有的性能提升和更稳定的体验。通过优化内存管理,降低了应用对系统资源的占用,确保设备…...
如何在 Ubuntu 22.04 上安装 Graylog 开源日志管理平台
简介 Graylog 的开源特性、丰富的功能、灵活性和可扩展性使其成为一个流行的日志管理平台。在本教程中,我将向你展示如何在 Ubuntu 22.04 上安装 Graylog,包括配置 Graylog 服务器软件包和访问 Graylog Web UI。 Graylog 是什么? Graylog …...
Wux weapp 组件库的 bug—— wux-picker选择器组件无法正确初始化到选定的value
options的value为Number,组件无法正常使用 解决方案,修改picker-view/utils.js中的getIndexFromValue函数,如下: export function getIndexFromValue(value, col [], fieldNames DEFAULT_FIELD_NAMES) {//return getRealIndex(…...
决策树(理论知识1)
目录 何为决策树决策树的组成决策树的构建 何为决策树 决策树(Decision Tree)是一种分类和回归方法,是基于各种情况发生的所需条件构成决策树,以实现期望最大化的一种图解法。由于这种决策 分支画成图形很像一棵树的枝干,故称决策树。它的运…...
【Spring】获取Bean对象需要哪些注解
阿华代码,不是逆风,就是我疯 你们的点赞收藏是我前进最大的动力!! 希望本文内容能够帮助到你!! 目录 一:Service(服务存储) 1:存储bean的代码 2࿱…...
java的Webclient对象怎解解析400状态码
在Java中使用WebClient处理400状态码,可以通过检查响应状态并根据状态码进行相应的错误处理。以下是几种处理400状态码的方法: 使用onStatus方法判断和处理错误: 你可以使用WebClient的retrieve()方法链中的onStatus方法来检查响应状态码。如…...
【计算机视觉基础CV-图像分类】03-深度学习图像分类实战:鲜花数据集加载与预处理详解
本文将深入介绍鲜花分类数据集的加载与处理方式,同时详细解释代码的每一步骤并给出更丰富的实践建议和拓展思路。以实用为导向,为读者提供从数据组织、预处理、加载到可视化展示的完整过程,并为后续模型训练打下基础。 前言 在计算机视觉的深…...
ubuntu 如何重装你的apt【apt-get报错: symbol lookup error/undefined symbol】
副标题:解决error:apt-get: symbol lookup error: /lib/x86_64-linux-gnu/libapt-private.so.0.0: undefined symbol: _ZNK13pkgTagSection7FindULLENS_3KeyERKy, version APTPKG_6.0 文章目录 问题描述报错分析解决方案:重装你的apt1、查看你的ubuntu版本2、下载适配你的ap…...
Unity 上好用的插件
PlayerMaker BehaviorDesigner Cinemachine Timeline Hybrid Addressable AssetBundle Blower Simple Zoom 大地图上缩放和平移使用ScrollRect的好效果实现...
大数据机器学习算法和计算机视觉应用07:机器学习
Machine Learning Goal of Machine LearningLinear ClassificationSolutionNumerical output example: linear regressionStochastic Gradient DescentMatrix Acceleration Goal of Machine Learning 机器学习的目标 假设现在有一组数据 x i , y i {x_i,y_i} xi,yi&…...
Godot RPG 游戏开发指南
Godot RPG 游戏开发指南 一、基础准备 1. 开发环境 下载并安装最新版 Godot 4.x选择使用 GDScript 或 C# 作为开发语言准备基础美术资源(角色、地图、道具等) 2. 项目结构 project/ ├── scenes/ # 场景文件 ├── scripts/ # 脚…...
c++ 找第一个只出现一次的字符
【题目描述】 给定一个只包含小写字母的字符串,请你找到第一个仅出现一次的字符。如果没有,输出no。 【输入】 一个字符串,长度小于100000。 【输出】 输出第一个仅出现一次的字符,若没有则输出no。 【输入样例】 abcabd【输出样…...
时空信息平台架构搭建:基于netty封装TCP通讯模块(IdleStateHandler网络连接监测,处理假死)
文章目录 引言I 异步TCP连接操作II 心跳机制:空闲检测(读空闲和写空闲)基于Netty的IdleStateHandler类实现心跳机制(网络连接监测)常规的处理假死健壮性的处理假死方案获取心跳指令引言 基于netty实现TCP客户端:封装断线重连、连接保持 https://blog.csdn.net/z92911896…...
【Rust自学】3.6. 控制流:循环
3.6.0. 写在正文之前 欢迎来到Rust自学的第三章,一共有6个小节,分别是: 变量与可变性数据类型:标量类型数据类型:复合类型函数和注释控制流:if else控制流:循环(本文) 通过第二章…...
如何正确计算显示器带宽需求
1. 对显示器的基本认识 一个显示器的参数主要有这些: 分辨率:显示器屏幕上像素点的总数,通常用横向像素和纵向像素的数量来表示,比如19201080(即1080p)。 刷新率:显示器每秒钟画面更新的次数&…...
mysql 基于chunk机制是如何支持运行期间,动态调整buffer pool大小的
mysql 基于chunk机制是如何支持运行期间,动态调整buffer pool大小的 MySQL 的 InnoDB 存储引擎确实支持在运行期间动态调整缓冲池(buffer pool)的大小,但其机制与自定义缓存系统有所不同。InnoDB 通过内部优化和配置参数来实现这…...
梳理你的思路(从OOP到架构设计)_简介设计模式
目录 1、 模式(Pattern) 是较大的结构编辑 2、 结构形式愈大 通用性愈小编辑 3、 从EIT造形 组合出设计模式 1、 模式(Pattern) 是较大的结构 组合与创新 達芬奇說:簡單是複雜的終極形式 (Simplicity is the ultimate form of sophistication) —Leonardo d…...
【Redis】缓存
什么是缓存 https://tech.meituan.com/2017/03/17/cache-about.html Spring Data Redis Spring Data Redis提供了从Spring应用程序轻松配置和访问Redis的功能。 引入依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>sp…...
基于 PyCharm 和 Navicat 的新闻管理系统
# 用于创建连接池 pip3 install mysql-connector-python # 改变终端打印颜色 pip3 install colorama 1.创建连接池 文件地址:db/mysql_db.py 首先建立一个与 MySQL 数据库的连接池,以便在应用程序中复用连接,提高性能。 如果连接池创建失败…...
<QNAP 453D QTS-5.x> 日志记录: 优化性能 内存管理 修改swap优先顺序 swap放在ssd 网络稳定性 进程出错管理
起因 几个月前,开始重学编程,往 NAS 的 docker 里放了些 containers ,每一个用来跑练习的 App。为了放更多的app,上个月加了 4GB 内存。最近只放了两个,NAS 就会时不时的闪断。codes 全存在网络驱动器上,当…...
一区牛顿-拉夫逊算法+分解+深度学习!VMD-NRBO-Transformer-GRU多变量时间序列光伏功率预测
一区牛顿-拉夫逊算法分解深度学习!VMD-NRBO-Transformer-GRU多变量时间序列光伏功率预测 目录 一区牛顿-拉夫逊算法分解深度学习!VMD-NRBO-Transformer-GRU多变量时间序列光伏功率预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 1.中科院一区…...
本地部署webrtc应用怎么把http协议改成https协议?
环境: WSL2 Ubuntu22.04 webrtc视频聊天应用 问题描述: 本地部署webrtc应用怎么把http协议改成https协议? http协议在安卓手机浏览器上用不了麦克风本,来地应用webrtc 本来是http协议,在安卓手机上浏览器不支持使…...
React简单了解
原理简化了解 import React from "react" import { createRoot } form "react-dom/client"const element React.createElement(p,{id: hello},Hello World! )const container document.querySelector(#root) const root createRoot(container) root.r…...
基于LabVIEW的USRP信道测量开发
随着无线通信技术的不断发展,基于软件无线电的设备(如USRP)在信道测量、无线通信测试等领域扮演着重要角色。通过LabVIEW与USRP的结合,开发者可以实现信号生成、接收及信道估计等功能。尽管LabVIEW提供了丰富的信号处理工具和图形…...
Docker挂载
目录 数据卷挂载 本地目录挂载 数据卷挂载 宿主机默认的存放所有容器数据卷的目录:/var/lib/docker/volumes nginx容器 静态文件目录:/usr/share/nginx/html 配置文件目录:/etc/nginx/nginx.conf 修改宿主机的内容,进入到容器查…...
使用Java结合经纬度位置计算目标点的日出日落时间
目录 前言 一、应用示例 1、天安门升旗时间 2、湖南省日出日落信息 二、JAVA日出日落计算 1、在线API 2、使用Java进行计算 三、总结 前言 随着城市化进程的加速,城市环境与人类生活的联系日益紧密。城市不仅承载着居住、工作、休闲等多种功能,也…...
八字精批api接口_php获取生成八字和批注的方法研究
八字算命 API 介绍 这个八字算命 API 提供了一种便捷的方式,让用户通过 GET 或 POST 请求获取详细的八字信息。API 返回的数据格式为 JSON,包含多种命理分析和建议,适合对传统命理学感兴趣的用户。 API 功能 五行分析: 提供用户…...
docker run 命令参数
user docker run -it --nameubn18 --gpus all --privilegedtrue --shm-size 8G ubuntu:18.04 /bin/bash-it 是什么意思 4o 在运行 docker run 命令时,-it 是两个选项的组合,用于更好地与容器进行交互: -i 或 --interactive:这个选…...
智能外呼技术如何改变企业营销方式
智能外呼技术如何改变企业营销方式 作者:开源大模型智能呼叫中心系统FreeAICC,Github:https://github.com/FreeIPCC/FreeAICC 在数字化时代,企业营销方式正经历着前所未有的变革。其中,智能外呼技术作为一项前沿的人…...
redis数据转移
可能有时候因为硬件的原因我们我们需要更换服务器,如果更换服务器的话,那我们redis的数据该怎样转移呢,按照一下步骤即可完成redis数据的转移 1.进入redis客户端 2.使用 bgsave命令进行数据的备份,此命令完成后会在你的redis安装目…...
STM32-笔记5-按键点灯(中断方法)
1、复制03-流水灯项目,重命名06-按键点灯(中断法) 在\Drivers\BSP目录下创建一个文件夹exti,在该文件夹下,创建两个文件exti.c和exti.h文件,并且把这两个文件加载到项目中,打开项目工程文件 加载…...
DotNetBrowser 3.0.0 正式发布!
🛠️ 重要消息:DotNetBrowser 3.0.0 正式发布! 我们很高兴向您介绍全新的 DotNetBrowser 3.0.0 版本。此次更新带来了多项重要功能与优化,进一步提升了 Web 开发的效率和体验。 📢 DotNetBrowser 3.0.0 包含哪些新功…...
MySQL基础笔记(三)
在此特别感谢尚硅谷-康师傅的MySQL精品教程 获取更好的阅读体验请前往我的博客主站! 如果本文对你的学习有帮助,请多多点赞、评论、收藏,你们的反馈是我更新最大的动力! 创建和管理表 1. 基础知识 1.1 一条数据存储的过程 存储数据是处理数…...
SEO初学者-搜索引擎如何工作
搜索引擎基础搜索引擎是如何建立索引的搜索引擎如何对网页进行排名搜索引擎是如何个性化搜索结果的 搜索引擎的工作方式是使用网络爬虫抓取数十亿个页面。爬虫也称为蜘蛛或机器人,它们在网络上导航并跟踪链接以查找新页面。然后,这些页面会被添加到搜索引…...
在Ubuntu下运行QEMU仿真FreeBSD riscv64系统
在Ubuntu下运行QEMU仿真FreeBSD riscv64系统 突发奇想,尝试在Ubuntu下运行QEMU仿真FreeBSD riscv64系统, 参考这篇文档:手把手教你在QEMU上运行RISC-V Linux_qemu 运行 .bin-CSDN博客 并参考FreeBSD的Wiki:riscv - FreeBSD Wik…...
SQL 使用带聚集函数的联结
聚集函数用于汇总数据,通常用于从一个表中计算统计信息,但也可以与联结一起使用。以下是一个例子,展示如何使用聚集函数统计每个顾客的订单数。 示例 1:使用 COUNT() 函数与 INNER JOIN 假设我们需要检索所有顾客及每个顾客所下…...
Linux网络功能 - 服务和客户端程序CS架构和简单web服务示例
By: fulinux E-mail: fulinux@sina.com Blog: https://blog.csdn.net/fulinus 喜欢的盆友欢迎点赞和订阅! 你的喜欢就是我写作的动力! 目录 概述准备工作扫描服务端有那些开放端口创建客户端-服务器设置启动服务器和客户端进程双向发送数据保持服务器进程处于活动状态设置最小…...
爬取Q房二手房房源信息
文章目录 1. 实战概述2. 网站页面分析3. 编写代码爬取Q房二手房房源信息3.1 创建项目与程序3.2 运行程序,查看结果 4. 实战小结 1. 实战概述 本次实战项目旨在通过编写Python爬虫程序,抓取深圳Q房网上的二手房房源信息。我们将分析网页结构,…...
【JavaEE初阶】线程 和 thread
本节⽬标 认识多线程 掌握多线程程序的编写 掌握多线程的状态 一. 认识线程(Thread) 1概念 1) 线程是什么 ⼀个线程就是⼀个 "执⾏流". 每个线程之间都可以按照顺序执⾏⾃⼰的代码. 多个线程之间 "同时" 执⾏着多份代码. 还…...
【IMU:视觉惯性SLAM系统】
视觉惯性SLAM系统简介 相机(单目/双目/RGBD)与IMU结合起来就是视觉惯性,通常以单目/双目IMU为主。 IMU里面有个小芯片可以测量角速度与加速度,可分为6轴(6个自由度)和9轴(9个自由度)IMU,具体的关于IMU的介…...
【python实现烟花】
可以使用 Python 的 turtle 模块来实现烟花效果。下面是一个简单的示例代码,展示如何用 turtle 绘制烟花: import turtle import random# 设置屏幕 screen turtle.Screen() screen.bgcolor("black")# 创建烟花函数 def draw_firework(x, y):…...
OpenCV学习——图像融合
import cv2 as cv import cv2 as cvbg cv.imread("test_images/background.jpg", cv.IMREAD_COLOR) fg cv.imread("test_images/forground.png", cv.IMREAD_COLOR)# 打印图片尺寸 print(bg.shape) print(fg.shape)resize_size (1200, 800)bg cv.resize…...
CS 144 check6: buiding an IP router
Lecture Notes Exercises 路由器的任务是根据路由表转发接收到的数据报:路由表是一系列规则,用于指导路由器针对任何给定的数据报应如何进行转发。 发送出什么接口。下一跳的IP地址。 这个check的工作是实现一个路由器,它可以为任何给定的…...
Pytorch | 从零构建ResNet对CIFAR10进行分类
Pytorch | 从零构建ResNet对CIFAR10进行分类 CIFAR10数据集ResNet核心思想网络结构创新点优点应用 ResNet结构代码详解结构代码代码详解BasicBlock 类ResNet 类ResNet18、ResNet34、ResNet50、ResNet101、ResNet152函数 训练过程和测试结果代码汇总resnet.pytrain.pytest.py 前…...
Spring整合Redis基本操作步骤
Spring 整合 Redis 操作步骤总结 1. 添加依赖 首先,在 pom.xml 文件中添加必要的 Maven 依赖。Redis 相关的依赖包括 Spring Boot 的 Redis 启动器和 fastjson(如果需要使用 Fastjson 作为序列化工具): <!-- Spring Boot Re…...
java中的方法的重载和重写、构造器
目录 方法的重载和重写、构造器1.java的修饰符:2.普通方法3.构造器(也叫构造方法/构造函数)4.方法的重载5.补充6.方法的重写7.类的执行顺序8.再看方法的重写 方法的重载和重写、构造器 1.java的修饰符: public修饰的代码…...