当前位置: 首页 > article >正文

go 语言中的线程池

使用 goroutine 和 channel

Go 语言中并没有直接类似 Java 线程池的内建概念,但它提供了类似的功能,主要通过goroutinechannel来实现并发处理。你可以通过结合这两者来实现一个“线程池”的功能。

在 Go 中,goroutine是轻量级的线程,它由 Go 的调度器管理,可以非常容易地启动并发任务。而channel则用于在不同 goroutine 之间传递消息或同步操作。

要实现一个类似线程池的功能,你可以:

  1. 创建一个固定数量的 goroutine 来处理任务。
  2. 通过 channel 将任务传递给这些 goroutine。
  3. 使用一个 pool 来管理 goroutine 的生命周期。

以下是一个简单的例子,模拟一个固定大小的“线程池”:

package mainimport ("fmt""sync"
)type Task struct {ID int
}type WorkerPool struct {tasks   chan Taskworkers intwg      sync.WaitGroup
}func (wp *WorkerPool) Start() {for i := 0; i < wp.workers; i++ {go wp.worker(i)}
}func (wp *WorkerPool) worker(workerID int) {defer wp.wg.Done()for task := range wp.tasks {// 处理任务fmt.Printf("Worker %d is processing task %d\n", workerID, task.ID)}
}func (wp *WorkerPool) AddTask(task Task) {wp.tasks <- task
}func (wp *WorkerPool) Close() {close(wp.tasks)wp.wg.Wait()
}func main() {// 创建一个有 3 个 worker 的池pool := WorkerPool{tasks:   make(chan Task, 10),workers: 3,}// 启动 workerpool.Start()// 添加任务for i := 1; i <= 5; i++ {pool.AddTask(Task{ID: i})}// 关闭并等待所有任务完成pool.Close()
}

在这个例子中:

  • WorkerPool 类似于一个线程池,管理多个 worker goroutine。
  • AddTask 用于将任务添加到任务队列(channel)。
  • Start 启动 worker goroutine 来处理任务。
  • Close 用于关闭任务队列并等待所有任务完成。

通过这种方式,你可以控制并发数量,避免创建过多的 goroutine,也能有效地管理任务执行。

如果你想要更灵活的线程池实现,Go 社区中也有一些第三方库,比如 ants,它提供了一个成熟的线程池实现,支持高效的资源管理和任务调度。
除了直接通过 goroutinechannel 来实现类似线程池的功能,Go 语言还有一些其他方式可以实现类似于 Java 中的线程池概念。常见的方式包括:

使用 sync.Pool

sync.Pool 是 Go 提供的一个内存池机制,通常用于对象复用。虽然它本质上并不是一个线程池,但可以用它来创建一个类似的对象池,可以有效地复用已经处理完的 goroutine 或者任务对象,从而减少创建和销毁对象的开销。

package mainimport ("fmt""sync"
)type Task struct {ID int
}func main() {var pool sync.Pool// 初始化 poolpool.New = func() interface{} {return &Task{}}// 从 pool 获取对象task := pool.Get().(*Task)task.ID = 1fmt.Printf("Processing task %d\n", task.ID)// 将对象归还给 poolpool.Put(task)
}

sync.Pool 主要用于复用对象,因此可以通过复用 Task 对象来减少垃圾回收的负担,但它并不提供真正的并发任务调度和执行的功能。因此,sync.Pool 更适合用来管理对象池,而不直接适用于线程池的实现。

使用第三方库(如 ants

Go 社区提供了很多成熟的第三方库来帮助实现类似 Java 线程池的并发任务管理。一个常见的库是 ants,它实现了一个高效的 goroutine 池。

通过使用 ants,你可以实现任务的并发执行和资源池管理,提供了更多的功能和性能优化。

package mainimport ("fmt""github.com/panjf2000/ants/v2"
)func main() {// 创建一个线程池,最多支持 10 个并发任务pool, _ := ants.NewPool(10)defer pool.Release()for i := 0; i < 20; i++ {task := ipool.Submit(func() {// 处理任务fmt.Printf("Processing task %d\n", task)})}
}

在这个例子中:

  • 使用 ants.NewPool 创建一个大小为 10 的线程池,最多可以同时处理 10 个任务。
  • 使用 pool.Submit 提交任务到线程池中。
  • 任务由池中的工作 goroutine 执行。

ants 库提供了线程池的管理,包括池大小、任务调度和资源释放等功能,比直接使用 goroutine 和 channel 更加方便和高效。

通过自定义调度器管理 goroutine

另一种方式是自定义一个调度器,它可以限制同时运行的 goroutine 数量,避免系统资源被过度消耗。例如,使用一个调度器队列来管理任务的执行。

package mainimport ("fmt""sync""time"
)type Task struct {ID int
}type Scheduler struct {taskQueue chan Taskwg        sync.WaitGroup
}func NewScheduler(workerCount int) *Scheduler {return &Scheduler{taskQueue: make(chan Task),}
}func (s *Scheduler) Start(workerCount int) {for i := 0; i < workerCount; i++ {go s.worker(i)}
}func (s *Scheduler) worker(workerID int) {for task := range s.taskQueue {// 处理任务fmt.Printf("Worker %d is processing task %d\n", workerID, task.ID)time.Sleep(time.Second) // 模拟任务执行时间s.wg.Done()}
}func (s *Scheduler) AddTask(task Task) {s.wg.Add(1)s.taskQueue <- task
}func (s *Scheduler) Close() {close(s.taskQueue)s.wg.Wait()
}func main() {scheduler := NewScheduler(3)scheduler.Start(3)// 提交任务for i := 1; i <= 10; i++ {scheduler.AddTask(Task{ID: i})}// 等待任务完成scheduler.Close()
}

在这个实现中:

  • Scheduler 使用 taskQueue 管理任务,限制了同时处理任务的 goroutine 数量。
  • worker 会从 taskQueue 中取任务,并处理它。
  • AddTask 用来提交任务,Close 用来关闭任务队列并等待所有任务完成。

这种方法允许你自定义更多的调度策略,控制任务的执行和 goroutine 的管理。

总结

Go 语言本身并没有类似 Java 线程池的直接概念,但你可以使用以下几种方式来实现类似功能:

  • 通过 goroutine 和 channel 手动实现线程池。
  • 使用 sync.Pool 管理对象池。
  • 使用社区库如 ants 来高效管理 goroutine 池。
  • 自定义调度器来限制并发任务数。

根据你的需求,选择合适的方式来实现并发任务的管理。

相关文章:

go 语言中的线程池

使用 goroutine 和 channel Go 语言中并没有直接类似 Java 线程池的内建概念&#xff0c;但它提供了类似的功能&#xff0c;主要通过goroutine和channel来实现并发处理。你可以通过结合这两者来实现一个“线程池”的功能。 在 Go 中&#xff0c;goroutine是轻量级的线程&…...

从 0 到 1,用 Python 构建超实用 Web 实时聊天应用

从 0 到 1&#xff0c;用 Python 构建超实用 Web 实时聊天应用 本文深入剖析如何运用 Python 的 Flask 框架与 SocketIO 扩展&#xff0c;搭建一个功能完备的 Web 实时聊天应用。从环境搭建、前后端代码实现&#xff0c;到最终运行展示&#xff0c;逐步拆解关键步骤&#xff0…...

AF3 DataPipeline类process_multiseq_fasta 方法解读

AlphaFold3 data_pipeline 模块DataPipeline类的 process_multiseq_fasta 方法用于处理多序列 FASTA 文件,生成 AlphaFold3 结构预测所需的特征,适用于多链复合物的预测。它结合了 Minkyung Baek 在 Twitter 上提出的“AlphaFold-Gap”策略,即通过在多链 MSA 中插入固定长度…...

Vue2+Element实现Excel文件上传下载预览【超详细图解】

目录 一、需求背景 二、落地实现 1.文件上传 图片示例 HTML代码 业务代码 2.文件下载 图片示例 方式一&#xff1a;代码 方式二&#xff1a;代码 3.文件预览 图片示例 方式一&#xff1a;代码 方式二&#xff1a;代码 一、需求背景 在一个愉快的年后&#xff…...

[记录贴] 火绒奇怪的进程保护

最近一次更新火绒6.0到最新版&#xff0c;发现processhacker的结束进程功能无法杀掉火绒的进程&#xff0c;弹窗提示如下&#xff1a; 可能是打开进程时做了权限过滤&#xff0c;火绒注册了两个回调函数如下&#xff1a; 但奇怪的是&#xff0c;在另外一台机器上面更新到最新版…...

【蓝桥杯】每天一题,理解逻辑(1/90)【Leetcode 移动零】

文章目录 题目解析讲解算法原理【双指针算法思路】(数组下标充当指针)如何划分和执行过程大致 代码详情 题目解析 题目链接&#xff1a;https://leetcode.cn/problems/move-zeroes/description/ 题目意思解析 把所有的零移动到数组的末尾保持非零元素的相对顺序 理解了这两层…...

vue js-web-screen-shot浏览器截取其他非全屏窗口界面

网页截屏 js-web-screen-shot 截取其他窗口 显示不全问题 npm 安装 js-web-screen-shot npm install js-web-screen-shot --savejs-web-screen-shot默认截屏是从左下角开始的&#xff0c;修改成左上角开始&#xff0c;然后编辑cropBoxInfo参数宽高进行截取&#xff0c;目前截…...

pycharm远程连接服务器运行pytorch

Linux部署pytorch 背景介绍 不同的开源代码可能需要不同的实验环境和版本&#xff0c;这时候的确体现出Anaconda管理环境的好处了&#xff0c;分别搞一个独立环境方便管理。 有的教程建议选择较旧的版本&#xff0c;但笔者建议在条件允许的情况下安装最新版&#xff0c;本次…...

服务器虚拟化是一种将物理服务器资源(如CPU、内存、存储、网络等)通过软件技术抽象、分割和整合,创建多个独立、隔离的虚拟服务器(虚拟机,VM)的技术。

服务器虚拟化是一种将物理服务器资源(如CPU、内存、存储、网络等)通过软件技术抽象、分割和整合,创建多个独立、隔离的虚拟服务器(虚拟机,VM)的技术。每个虚拟机可以运行不同的操作系统和应用程序,如同独立的物理服务器一样工作。 核心思想 资源池化:将物理服务器的硬…...

java练习(41)

ps&#xff1a;题目来自力扣 最接近的三数之和 给你一个长度为 n 的整数数组 nums 和 一个目标值 target。请你从 nums 中选出三个整数&#xff0c;使它们的和与 target 最接近。 返回这三个数的和。 假定每组输入只存在恰好一个解。 import java.util.Arrays;class Solut…...

关于CanvasRenderer.SyncTransform触发调用的机制

1&#xff09;关于CanvasRenderer.SyncTransform触发调用的机制 2&#xff09;小游戏Spine裁剪掉帧问题 3&#xff09;Dedicated Server性能问题 4&#xff09;.mp4视频放入RT进行渲染的性能分析闭坑指南 这是第421篇UWA技术知识分享的推送&#xff0c;精选了UWA社区的热门话题…...

【计算机网络】OSI模型、TCP/IP模型、路由器、集线器、交换机

一、计算机网络分层结构 计算机网络分层结构 指将计算机网络的功能划分为多个层次&#xff0c;每个层次都有其特定的功能和协议&#xff0c;并且层次之间通过接口进行通信。 分层设计的优势&#xff1a; 模块化&#xff1a;各层独立发展&#xff08;如IPv4→IPv6&#xff0c…...

PDF扫描档智能方向识别:多模型投票机制的实践测试 救活古典书籍

2025-02-22 20:10物联全栈123 尊敬的诸位&#xff01;我是一名物联网工程师。关注我&#xff0c;持续分享最新物联网与AI资讯和开发实战。期望与您携手探寻物联网与 AI 的无尽可能 RAG知识库搭建的过程中&#xff0c;扫描档pdf的支持和准确率一直是个大家都不愿主动提起的事情…...

java23种设计模式-桥接模式

桥接模式&#xff08;Bridge Pattern&#xff09;学习笔记 &#x1f31f; 定义 桥接模式属于结构型设计模式&#xff0c;将抽象部分与实现部分分离&#xff0c;使它们可以独立变化。通过组合代替继承的方式&#xff0c;解决多维度的扩展问题&#xff0c;防止类爆炸。 &#x…...

【11】RUST使用cargo组织crate

文章目录 使用cargo组织crate重导出编译文档生成测试 cargo组织工作空间 TODOcrate.io账号 TODO暂时不看发布crate 使用cargo组织crate 重导出 在模块顶部使用pub use self::重导出&#xff0c;方便使用模块时候直接使用use mod_X::xxx。从而隐藏crate内部模块的结构。方便向…...

springboot 引入前端

前端 打包 npm run build vue.config.js 文件 publicPath 默认建议保持 / publicPath: ‘/’ 后端 目录 粘贴下面目录之一&#xff1a; src/main/resources/static/ src/main/resources/public/ 补充&#xff08;用的少&#xff09; server:servlet:context-path: /thirdAdm…...

ubuntu20.04 使用nmcli 连接wifi,并且设置永久连接

在 Ubuntu 20.04 中&#xff0c;你可以使用 nmcli 命令行工具来连接 WiFi 并设置为永久连接。以下是具体步骤&#xff1a; 1. 查看可用的 WiFi 网络 首先&#xff0c;使用以下命令查看可用的 WiFi 网络&#xff1a; nmcli dev wifi 这将列出所有可用的 WiFi 网络及其 SSID。…...

Android-创建mipmap-anydpi-v26的Logo

利用 Android Studio 自动创建 创建新项目&#xff1a;打开 Android Studio&#xff0c;点击 “Start a new Android Studio project” 创建新项目。在创建项目的过程中&#xff0c;当设置Target SDK Version为 26 或更高版本时&#xff0c;Android Studio 会在项目的res目录下…...

轻松搭建:使用Anaconda创建虚拟环境并在PyCharm中配置

一、使用Anaconda创建虚拟环境 1. 安装Anaconda 2..conda常用的命令 3. 创建虚拟环境-以搭建MachineVision为例 4. 激活虚拟环境 5. 安装依赖包 二、PyCharm配置环境 在进行Python项目开发时&#xff0c;合理的环境管理是必不可少的&#xff0c;特别是当你在多个项目中…...

驱动开发系列39 - Linux Graphics 3D 绘制流程(二)- 设置渲染管线

一:概述 Intel 的 Iris 驱动是 Mesa 中的 Gallium 驱动,主要用于 Intel Gen8+ GPU(Broadwell 及更新架构)。它负责与 i915 内核 DRM 驱动交互,并通过 Vulkan(ANV)、OpenGL(Iris Gallium)、或 OpenCL(Clover)来提供 3D 加速。在 Iris 驱动中,GPU Pipeline 设置 涉及…...

结构型模式 - 代理模式 (Proxy Pattern)

结构型模式 - 代理模式 (Proxy Pattern) 代理模式是一种结构型设计模式&#xff0c;它允许通过代理对象来控制对另一个对象&#xff08;目标对象&#xff09;的访问。代理对象充当目标对象的接口&#xff0c;客户端通过代理对象间接访问目标对象。 分为两大类 静态代理&#…...

Ubuntu 24.04 登录禁用用户列表

Ubuntu 24.04 登录禁用用户列表 得到登录用户列表状态禁用登录用户列表 得到登录用户列表状态 gsettings get org.gnome.login-screen disable-user-list禁用登录用户列表 gsettings get org.gnome.login-screen disable-user-list true...

MinIO整合SpringBoot实现文件上传、下载

文章目录 配置1. 部署MinIO服务2. 整合SpringBoot 功能实现1. 文件上传2. 文件下载 总结 配置 1. 部署MinIO服务 这里以docker为例&#xff1a; 安装minio命令docker run -p 9000:9000 -p 9001:9001 \ --name minio \ -v /path/to/data:/data \ -e "MINIO_ROOT_USERmin…...

【Python爬虫(90)】以Python爬虫为眼,洞察金融科技监管风云

【Python爬虫】专栏简介&#xff1a;本专栏是 Python 爬虫领域的集大成之作&#xff0c;共 100 章节。从 Python 基础语法、爬虫入门知识讲起&#xff0c;深入探讨反爬虫、多线程、分布式等进阶技术。以大量实例为支撑&#xff0c;覆盖网页、图片、音频等各类数据爬取&#xff…...

FreeRTOS(3)列表List

在 FreeRTOS 的源码中大量地使用了列表和列表项&#xff0c;因此想要深入学习 FreeRTOS&#xff0c;列表和列表项是必备的基础知识。这里所说的列表和列表项&#xff0c;是 FreeRTOS 源码中 List 和 List Item 的 直译&#xff0c;事实上&#xff0c; FreeRTOS 中的列表和列表项…...

C++和OpenGL实现3D游戏编程【连载23】——几何着色器和法线可视化

欢迎来到zhooyu的C++和OpenGL游戏专栏,专栏连载的所有精彩内容目录详见下边链接: 🔥C++和OpenGL实现3D游戏编程【总览】 1、本节实现的内容 上一节课,我们在Blend软件中导出经纬球模型时,遇到了经纬球法线导致我们在游戏中模型光照显示问题,我们在Blender软件中可以通过…...

Harmony开发笔记(未完成)

一、感想 作为一名拥有11年经验的Android开发者&#xff0c;我亲历了Android从高速发展到如今面临“僧多粥少”的过程。技术的世界瞬息万变&#xff0c;没有一种技术能够让人依赖一辈子。去年初&#xff0c;我自学了鸿蒙系统&#xff0c;并顺利通过了鸿蒙官方的初级和高级认。…...

【Java面试】创建线程有哪几种方式

目录 1.继承Thread类 2.实现Runnable接口 3.实现Callable接口和FutureTask 4.使用Executor框架&#xff08;线程池&#xff09; Java并发编程中不同接口和类之间的关系 总结 1.继承Thread类 优点&#xff1a; 简单直观。直接继承Thread类&#xff0c;可以方便地使用Threa…...

在Linux环境下利用MTCNN进行人脸检测(基于ncnn架构)

概述 本文将详细介绍如何在Linux环境下部署MTCNN模型进行人脸检测&#xff0c;并使用NCNN框架进行推理。 1. CMake的安装与配置 下载CMake源码 前往CMake官网下载&#xff0c;找到适合您系统的最新版本tar.gz文件链接&#xff0c;或者直接通过wget下载&#xff1a;CMake官方…...

AI数字人系统源码部署解决方案!!!

一、开场白 如今&#xff0c;科技的步伐越来越快&#xff0c;数字人已经从想象中走进了我们的现实生活。它们在娱乐、教育、医疗等多个领域大放异彩。了解数字人的代码开发技术&#xff0c;能让我们更好地理解其工作原理&#xff0c;为那些想在这一领域大展拳脚或者用数字人技…...