千锋教育-做有情怀、有良心、有品质的职业教育机构

当前位置:首页  >  IT面试题  >  Python面试题  >  正文

最新开源:高效的Python通用对象池化库

来源:千锋教育
发布时间:2022-09-26 15:19:00
分享

  高效的Python通用对象池化库

  在程序设计中,创建物体模块主要是通过生成对象来实现。当对象使用结束后,则会成为不再需要的模块进行销毁。

  而在系统进行对象的生成与销毁过程中会大量的增加内存的消耗,同时对象的销毁往往会留下残留的信息,这样将会伴随内存泄露的问题存在。

  在实际的程序开发过程中,往往需要生成和销毁大量重复的对象,这就使得内存泄漏产生的信息过多而无法被系统回收,从而占用系统更多的内存,而且生成物体过多时无法确定被什么模块实例化实现,对系统造成负担,不利于管理及后续操作,长此以往最终将导致程序变慢甚至崩溃。

  对象池是存放了一批已经创建好的对象的池,它是一个用来维护对象的结构。当程序需要使用对象的时候,可以直接从池中获取该对象,而不是实例化一个新的对象。

  在程序设计过程中,大部分人关注的往往只是对象的使用和效果的实现,实际上创建和使用之间还有一个初始化的过程,不过系统会将初始化和创建这两步结合在了一起,这样使得设计者忽略了系统创建和销毁对象这一过程对系统的影响。

  通常来讲,一个对象的创建和销毁过程开销很小,可以忽略不计,但是如果一个程序中涉及到一种对象多次创建,并且创建时间比较长,那就会能很明显的感觉到这部分的消耗所造成的系统速度受限。

  对象池可以看作是减少 GC 压力的首选方法,同时也是最简单的方法。

  对象池模式主要适用于以下应用场景:

  ●资源受限的场景。比如,不需要可伸缩性的环境(CPU、内存等物理资源有限),CPU性能不够强劲,内存比较紧张,垃圾收集,内存抖动会造成比较大的影响,需要提高内存管理效率, 响应性比吞吐量更为重要。

  ●在内存中数量受限的对象。

  ●创建成本高的对象。

  ●大量的存活期短且初始化成本低的对象池化,以降低内存分配和再分配成本,避免内存碎片。

  ●Python 的这样的动态语言,GC 是依靠引用技术来来保证对象不会过早的回收,某些场景下可能出现虽然创建了但是没人使用的空闲期,导致对象被回收了。可以委托给对象池来保管。

  Pond 介绍

  Pond 是一个 Python 中高效的通用对象池,具有性能好、内存占用小、命中率高的特点。基于近似统计的根据频率自动回收的能力,能够自动调整每个对象池的空闲对象数量。

  因为目前 Python 目前没有比较好的、测试用例完备、代码注释完备、文档完善的对象池化库,同时目前的主流对象池库也没有比较智能的自动回收机制。

  Pond 可能是 Python 中第一个社区公开的测试用例完整,覆盖率 90% 以上、代码注释完备、文档完善的对象池化库。

  Pond 灵感来自于 Apache Commons Pool、Netty Recycler、HikariCP、Caffeine,集合了多家的优点。

  其次 Pond 通过使用近似计数的方式以极小的内存空间统计每个对象池的使用频率,并且自动回收。

  流量较为随机平均的情况下,默认策略和权重可以降低 48.85% 内存占用,借取命中率 100%。

  8000700060005000400030002000NO POLICYDEFAULT POLICY 0.5N1000DEFAULT POLICY 0.8N403020-110

高效的Python通用对象池化库1

  流量较为符合 2/8 定律的情况下,默认策略和权重可以降低 45.7% 内存占用, 借取命中率 100%。

  8000700060005000400030002000NO POLICY10002/8 DEFAULT POLICY 0.5N2/8 DEFAULT POLICY 0.8N-1201040

高效的Python通用对象池化库2

  设计概述

  Pond 主要由 FactoryDict、Counter、PooledObjectTree 三部分以及一个单独的回收线程构成。

  FactoryDict

  使用 Pond 需要实现对象工厂 PooledObjectFactory,PooledObjectFactory 提供对象的创建、初始化、销毁、验证等操作,由 Pond 调用。

  所以为了让对象池支持存放完全不同的对象,Pond 使用了一个字典来记录每个工厂类的名称和自己实现的工厂类的实例化对象。

  每个 PooledObjectFactory 应该具备创建对象、销毁对象、验证对象是否还可用、重置对象四个功能。

  比较特别的是 Pond 支持自动重置对象,因为某些场景下可能会存在对象中要先赋值进行传递,传递完又被回收的情况,为了避免污染建议这种场景下无比实现这个功能。

  Counter

  Counter 中保存了一个近似计数器。

  PooledObjectTree

  PooleedObjectTree 是个字典,每个 key 对应着一个先进先出的队列,这些队列都是线程安全的。

  每个队列中保存着多个 PooleedObject。PooledObejct 保存了创建时间、最后借出的时间以及实际需要的对象。

  线程安全

  Pond 的借用和回收都是线程安全的。Python 的 queue 模块提供了一个适用于多线程编程的先进先出(FIFO)数据结构。它可以用来安全地在生产者和消费者线程之间传递消息或其他数据。

  锁是调用者来处理的,所有多个线程能够安全且容易的使用同样的 Queue 实例工作。而 Pond 的借用和回收都是在操作 queue,所以基本可以认为是线程安全的。

  借出机制

  在使用 Pond 借出一个对象时,会先检查想要借出的对象的种类是否已经在 PooledObjectTree 存在,如果存在会检查这个对象的对象池是否为空,如果为空会创建一个新的。

  如果对象池中有多余的对象,会利用 queue 弹出一个对象并验证这个对象是否可用。如果不可用会自动调用所属的 Factory 清理销毁该对象,同时清理它在 Python 中的 GC 计数,让它更快被 GC 回收,不断拿取下一个直至有可用的。

  如果这个对象可用,则会直接返回。当然无论是从对象池中取出对象还是新创建了一个对象,都会利用 Counter 增加一个计数。

  回收机制

  回收一个对象时会判断目标对象池存不存在,如果存在会检查对象池是否已经满了,满了的话会自动销毁要归还的这个对象。

  然后会检查这个对象是否已经被借出太长时间,如果超过了配置的最长时间同样会被清理掉。

  自动回收

  自动回收时每隔一段时间,默认是 300 s,就会执行一次。自动清理不经常使用的对象池中的对象。

  使用说明

  你可以先安装 Pond 的库并且在你的项目中引用。

高效的Python通用对象池化库3

  首先你需要声明一个你想要放入的类型的对象的工厂类,比如下面的例子我们希望池化的对象是 Dog,所以我们先声明一个 PooledDogFactory 类,并且实现 PooledObjectFactory。

高效的Python通用对象池化库4

  接着你需要创建 Pond 的对象:

高效的Python通用对象池化库5

  Pond 可以传递一些参数进去,分别代表:

  borrowed_timeout :单位为秒,借出对象的最长期限,超过期限的对象归还时会自动销毁不会放入对象池。

  time_between_eviction_runs :单位为秒,自动回收的间隔时间。

  thread_daemon :守护线程,如果为 True,自动回收的线程会随着主线程关闭而关闭。

  eviction_weight :自动回收时权重,会将这个权重与最大使用频次想乘,使用频次小于这个值的对象池中的对象都会进入清理步骤。

  实例化工厂类:

  factory = PooledDogFactory(pooled_maxsize=10, least_one=False)

  所有继承了 PooledObjectFactory 都会自带构造函数,可以传递 pooled_maxsize 和 least_one 两个参数。

  pooled_maxsize:这个工厂类生成出的对象的对象池的最大能放置的数量。

  least_one:如果为 True,在进入自动清理时,这个工厂类生成出的对象的对象池会至少保留一个对象。

  向 Pond 注册这个工厂对象,默认会使用 factory 的类名作为 PooledObjectTree 的 key :

  pond.register(factory)

  当然你也可以自定义它的名字,名字会作为 PooledObjectTree 的 key:

  pond.register(factory, name="PuppyFactory")

  注册成功后,Pond 会自动根据 factory 中设置的 pooled_maxsize 自动开始创建对象直至填满这个对象池。

  借用和归还对象:

高效的Python通用对象池化库6

  当然你可以用名字来进行借用和归还:

高效的Python通用对象池化库7

  完全清理一个对象池:

  pond.clear(factory)

  通过名字清理一个对象池:

  pond.clear(name="PuppyFactory")

  正常情况下,你只需要使用上面的这些方法,生成对象和回收对象都是全自动的。

  希望本篇文章可以给大家带来收获,如果喜欢的话,欢迎关注+点赞+收藏哦!后面会继续分享更多Python技术知识的!

声明:本站稿件版权均属千锋教育所有,未经许可不得擅自转载。

相关推荐

  • 最新开源:高效的Python通用对象池化库 接着你需要创建 Pond 的对象: Pond 可以传递一些参数进去,分别代表: borrowed_timeout :单位为秒,借出对象的最长期限,超过期限的对象归还时会自动销毁不会放入对象池。
  • 那些隐藏的Pycharm实用小技巧(下) 方法:点击左上角“File”→点击“Settings”→找到Project下面的“Python interpreter”→点击“+”→输入你想安装的包的名字,例如“pygame”,然后点击下面的“Install Package”就开始安装了,安装完成就可以用了。
  • 那些隐藏的Pycharm实用小技巧(上) 快速多行注释或者取消多行注释 虽然我们在写代码中可以用“#”来进行单行注释,但如果有多行代码需要注释,那么一行一行来敲“#”就有点麻烦,这里我们可以用鼠标选中多行代码,然后按 Ctrl+/ 就能进行多行代码的注释了,同时也可以取消多行代码的注释。
  • 关于框架的选择误区 在框架的选择问题上,许多人很容易就陷入了下面两个误区中而不自知:哪个框架最好——世上没有最好的框架,只有最适合你自己、最适合你的团队的框架。编程语言选择也是一个道理,你的团队Python最熟就用Python好了
  • web开发的一些框架 Django是一个开源的Web应用框架,由Python写成,支持许多数据库引擎,可以让Web开发变得迅速和可扩展,并会不断的版本更新以匹配Python最新版本,如果是新手程序员,可以从这个框架入手。
  • python库之文本处理 threading – Python标准库的线程运行。对于I/O密集型任务很有效。对于CPU绑定的任务没用,因为python GIL。 multiprocessing – 标准的Python库运行多进程。 celery – 基于分布式消息传递的异步任务队列/作业队列。