golang 连接池

news/2025/1/14 0:21:56/文章来源:https://www.cnblogs.com/timevalue/p/18669677

在使用 Go 语言开发高性能应用程序时,垃圾回收(GC)对性能的影响是一个不可忽视的问题。Go 的自动垃圾回收机制虽然方便,但其 Stop-The-World(STW)特性会导致程序暂停,影响性能。此外,频繁在堆上创建大量对象会增加垃圾回收的标记时间,进一步降低程序效率。因此,在性能优化时,采用对象池是一种有效的策略。通过对象池,可以回收不再使用的对象,避免它们被垃圾回收,从而在需要时无需重新在堆上创建对象。

除了对象池,管理数据库连接和 TCP 长连接也是提升性能的关键。这些连接的创建过程非常耗时,如果每次使用都重新创建连接,会大大增加业务处理时间。通过保存并复用这些连接,不仅可以减少业务耗时,还能显著提高应用程序的整体性能。

对象池包含一组已初始化且可重复使用的对象。用户可以从池中获取对象进行操作,并在使用完毕后归还对象,而不是直接销毁。当对象初始化成本高、实例化频繁但数量较少时,使用对象池可以显著提高性能。从池中获取对象的时间是可预测且较短的,而新建实例的时间则不确定,可能较长。

尽管对象池有诸多优点,但也存在一些潜在问题。例如,池中的对象可能会被垃圾回收,这对于数据库长连接等场景是不合适的。因此,在使用对象池时,需要确保对象的生命周期得到妥善管理,避免不必要的垃圾回收。

sync Pool 本身是线程安全,多个 goroutine 可以并发调用它的方法存取对象
sync.Pool 不可在使用之后再复制使用,它是有状态的对象,
sync.Pool 提供三个方法 New Get Put

New 创建池化对象
定义创建池化对象的方法,这样可以在有需要的还还好创建对象,当调用 Pool 的 Get方法从池中获取对象时候,没有空闲对象可用时候,会调用 New 方法创建新的对象,如果没有设置 New 字段则 没空闲对象可用时候, Get方法将返回 nil

Get 获取对象
调用 Get 方法,就会从池子取走一个对象,返回给调用者 ,也有可能是返回一个 nil (如果 Pool.New没设置,Pool里面没有空对象的情况下)

Put 返还对象
Put 方法将一个对象返回给 Pool,Pool 会把这个对象保存到池子中,并且可以复用,如果返还的对象是 nil, Pool 会忽略,如果你想弃用一个对象,不再重用,很简单,不要调用 Put 方法返还即可

package mainimport ("fmt""net/http""sync""time"
)func main() {var p sync.Poolp.New = func() any {return &http.Client{Timeout: 5 * time.Second,}}var wg sync.WaitGroupwg.Add(100)t1 := time.Now()go func() {for i := 0; i < 100; i++ {go func() {defer wg.Done()c := p.Get().(*http.Client)defer p.Put(c)resp, err := c.Get("https://bing.com")if err != nil {fmt.Println("err:", err)return}resp.Body.Close()fmt.Print("got bing.com")}()}}()wg.Wait()fmt.Println("cost", time.Since(t1))}

使用 sync.Pool 可以池化任意对象,经常用它池化 byte slice,创建一个字节池,避免频繁创建销毁 byte slice

github.com/PuerkitoBio/bytebufferpool 功能与 sync.Pool 相同,底层也是 sync.Pool ,
改进之处有
1、动态调整缓冲区大小:bytebufferpool 通过将缓冲区大小分为20个区间,根据放回对象的容量落在哪个区间来记录次数。当放回次数达到一定阈值时,会重新校准,计算哪个区间容量的对象最多,并将 defaultSize 设置为该区间的上限容量。这样可以避免在使用过程中的切片扩容,从而提升性能。:
2、防止内存碎片化:
通过限制最大缓冲区大小,防止过大的缓冲区占用过多内存,导致内存碎片化和GC异常。maxSize 参数可以动态调整,确保内存使用在合理范围内。

github.com/oxtoacart/bpool

bpool 设计的精妙之处

  1. 减少内存分配和回收

    • bpool 通过创建一个固定数量的预分配缓冲区,减少了内存分配和回收的次数。这种机制显著提高了内存操作的效率,尤其是在高并发环境下。
  2. 智能缓冲区管理

    • SizedBufferPool 提供了一种智能策略,可以限制缓冲区的最大增长,防止过度使用的内存占用。这有助于避免内存泄漏和过度消耗系统资源。
  3. 多种缓冲池类型

    • bpool 提供了三种类型的缓冲池:
      • BufferPool:提供固定大小的 bytes.Buffer 缓冲池。
      • BytePool:提供固定长度的 []byte 切片池。
      • SizedBufferPool:与 BufferPool 类似,但会预先设定缓冲区的容量,并在返回时丢弃增长过大的缓冲区。

sync.Pool 使用陷阱

1.对象重用不当 sync.Pool 的主要目的是重用对象,但如果对象状态未正确重置,可能会导致不可预期的行为。
2. .忽略对象的生命周期
sync.Pool 中的对象可能在任何时候被垃圾回收器回收,因此不应依赖其持久性。
3. 并发安全问题
问题:虽然 sync.Pool 本身是线程安全的,但用户在获取对象后修改其状态时,必须小心其他 goroutine 可能会同时获取相同的对象。
4. 不适合长期存活的对象
问题:sync.Pool 设计用于临时对象,不适合用于长期存在的对象。解决方法:对于需要长期存活的对象,如数据库连接或 TCP 连接,应使用专门的连接池管理,而不是 sync.Pool。确保在使用完对象后正确关闭或释放资源。
5. 监控和调整池子大小
问题:sync.Pool 的大小是动态的,但如果不监控其使用情况,可能会导致资源浪费或不足。
比如说 底层是一个切片数组,取出来切片 buffer,往这个里面添加了大量数据,导致底层数组切片很大,即使调用 bytes.Buffer 的Reset方法 然后将其放回池子,底层容量没有改变,所占空间仍然很大。由于 pool 的回收机制,这些大 buffer 可能不被回收,一直占用很大空间,从而造成内存泄漏,浪费。解决办法是,将元素放回时候,增加检查逻辑,如果返还元素超过一定大小的 buffer,直接丢弃,不再放回池子。

tcp连接池,常用库
github.com/faith/pool

package mainimport ("context""fmt""log""net""github.com/fatih/pool"
)type TCPConnection struct {conn net.Conn
}func (conn *TCPConnection) Reset() {conn.conn.Close()conn.conn = nil
}func newTCPConnection(address string) (*TCPConnection, error) {conn, err := net.Dial("tcp", address)if err != nil {return nil, err}return &TCPConnection{conn: conn}, nil
}func main() {address := "localhost:8080"p := pool.New(func() interface{} {conn, err := newTCPConnection(address)if err != nil {log.Println("Failed to create new connection:", err)return nil}return conn})defer p.Close()for i := 0; i < 10; i++ {obj := p.Get()if obj == nil {log.Println("Failed to get connection from pool")continue}conn := obj.(*TCPConnection)// 使用连接_, err := conn.conn.Write([]byte("Hello, server!"))if err != nil {log.Println("Failed to write to connection:", err)continue}buffer := make([]byte, 1024)n, err := conn.conn.Read(buffer)if err != nil {log.Println("Failed to read from connection:", err)continue}fmt.Println("Received:", string(buffer[:n]))// 释放连接p.Put(obj)}
}

任务池
github.com/gammazero/workerpool
github.com/ivpusic/grpool

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

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

相关文章

快速上手 INFINI Console 的 TopN 指标功能

背景 在分布式搜索引擎系统(如 Easysearch、Elasticsearch 和 OpenSearch)中,性能监控至关重要。为了确保系统的高效运行和资源的合理分配,我们通常需要关注一段时间内关键资源的使用情况,特别是索引、节点和分片的内存与 CPU 占用情况。 通过对这些关键指标进行 TopN 查询…

特斯拉CEO埃隆马.斯克的五步工作法,怎么提高工程效率加速产品开发?

简介 在《埃隆马斯克传》这本书中,有两个章节写到了特斯拉 CEO 埃隆马斯克为了在一段时间内,提升特斯拉汽车 model 3 的产能到每个月 5000 辆这个数量级,在书中叫 “量产地狱”,这是他的目标。 马斯克扎根工厂,睡在工厂的地板上近一年,亲自参与生产线的调试和优化,通过反…

iStoreOS_x86-U盘安装写入内置硬盘启动

https://doc.linkease.com/zh/guide/istoreos/install_x86.html#_1-%E8%A7%86%E9%A2%91%E4%BB%8B%E7%BB%8D 86 物理机x86物理机,范围很广,可以是各种"电脑",或者J4125/N5105等小主机。这里介绍x86实机安装iStoreOS固件。#1.视频介绍#2.准备工作一个 U盘 一个显示器…

《CPython Internals》阅读笔记:p118-p150

《CPython Internals》学习第 8 天,p118-p150 总结,总计 33 页。 一、技术总结 补充一些本人整理的关于 Context-Free Grammar(CFG) 的知识。 1.symbol(符号) A mathematical symbol is a figure or a combination of figures that is used to represent a mathematical ob…

基于遗传优化的货柜货物摆放优化问题求解matlab仿真

1.程序功能描述 基于遗传优化的货柜货物摆放优化问题求解matlab仿真。在一个货架上,初始状态下,随机将货物放在货柜上,优化之后,整理输出整理后的货物摆放效果。 2.测试软件版本以及运行结果展示MATLAB2022A版本运行 3.核心程序for ij = 1:Iterij%适应度yfit = 1./(…

【MSF免杀】python木马源码免杀

免责声明 由于传播利用本文所提供的信息而造成的任何直接或者间接的后果及损失,均由使用者本人负责,一旦造成后果请自行承担!🧲【MSF代码审计】Java木🐎源码分析这篇写了 MSF 的木马分析,今天就来实现一下免杀吧 。 开启免杀之路 java 代码量太大了,几百行代码 python…

Proj CJI Paper Reading: A False Sense of Safety: Unsafe Information Leakage in Safe AI Responses

Abstract 本文:Tasks:Decomposition Attacks: get information leakage of LLMMethod: 利用LLM(称为ADVLLM)+Few shots example把一个恶意的问题分成许多小的问题,发送给Victim LLMs,再使用ADVLLM把这些问题的回答拼凑出来得到答案 拆分原则是最大化与impermissible informat…

基于遗传优化的Sugeno型模糊控制器设计matlab仿真

1.课题概述基于遗传优化的Sugeno型模糊控制器设计matlab仿真,通过遗传优化算法优化模糊控制器的隶属函数参数,从而获得较优的控制效果。2.系统仿真结果 3.核心程序与模型 版本:MATLAB2022aMAXGEN = 15; NIND = 10; Nums = 1; Chrom =crtbp(NIND,Nums*10);%sh Areas = …

tensor学习

tensor 汇总都练习

GIF图片网

https://tenor.com/

Pytorch深入学习快速入门--小土堆

第五节:Pytorch加载数据初识 Dataset :提供一种方式获取数据及其lebel 本例子中ants、beer就是lebel 例子:from torch.utils.data import Dateset

使用GTD工作法提升效率

前言 近年来随着工作、副业的开展,每天要做的事情越来越多,而且还积攒了很多工作,每天大脑被各种事情充斥着,乱糟糟的,不仅效率很低,还很容易导致焦虑。 为此我一直有在寻找合适的项目管理工具,也看了一些相关的书籍,不过很多方法都复杂且难以快速实践。 直到我上网冲浪…