做者按:《天天一个设计模式》旨在初步领会设计模式的精髓,目前采用
javascript
和python
两种语言实现。诚然,每种设计模式都有多种实现方式,但此小册只记录最直截了当的实现方式 :)javascript
我的技术博客-godbmw.com 欢迎来玩! 每周至少 1 篇原创技术分享,还有开源教程(webpack、设计模式)、面试刷题(偏前端)、知识整理(每周零碎),欢迎长期关注!本篇博客地址是:《天天一个设计模式之享元模式》。前端
若是您也想进行知识整理 + 搭建功能完善/设计简约/快速启动的我的博客,请直接戳theme-bmwjava
享元模式:运用共享技术来减小建立对象的数量,从而减小内存占用、提升性能。python
享元模式虽然名字听起来比较高深,可是实际使用很是容易:只要是须要大量建立重复的类的代码块,都可以使用享元模式抽离内部/外部状态,减小重复类的建立。webpack
为了显示它的强大,下面的代码是简单地实现了你们耳熟能详的“对象池”,以彰显这种设计模式的魅力。git
这里利用python
和javascript
实现了一个“通用对象池”类--ObjectPool
。这个类管理一个装载空闲对象的数组,若是外部须要一个对象,直接从对象池中获取,而不是经过new
操做。es6
对象池能够大量减小重复建立相同的对象,从而节省了系统内存,提升运行效率。github
为了形象说明“享元模式”在“对象池”实现和应用,特别准备了模拟了File
类,而且模拟了“文件下载”操做。web
经过阅读下方代码能够发现:对于File
类,内部状态是pool
属性和download
方法;外部状态是name
和src
(文件名和文件连接)。借助对象池,实现了File
类的复用。面试
注:为了方便演示,Javascript
实现的是并发操做,Python
实现的是串行操做。输出结果略有不一样。
from time import sleep class ObjectPool: # 通用对象池 def __init__(self): self.__pool = [] # 建立对象 def create(self, Obj): # 对象池中没有空闲对象,则建立一个新的对象 # 对象池中有空闲对象,直接取出,无需再次建立 return self.__pool.pop() if len(self.__pool) > 0 else Obj(self) # 对象回收 def recover(self, obj): return self.__pool.append(obj) # 对象池大小 def size(self): return len(self.__pool) class File: # 模拟文件对象 def __init__(self, pool): self.__pool = pool def download(self): # 模拟下载操做 print('+ 从', self.src, '开始下载', self.name) sleep(0.1) print('-', self.name, '下载完成') # 下载完毕后,将对象从新放入对象池 self.__pool.recover(self) if __name__ == '__main__': obj_pool = ObjectPool() file1 = obj_pool.create(File) file1.name = '文件1' file1.src = 'https://download1.com' file1.download() file2 = obj_pool.create(File) file2.name = '文件2' file2.src = 'https://download2.com' file2.download() file3 = obj_pool.create(File) file3.name = '文件3' file3.src = 'https://download3.com' file3.download() print('*' * 20) print('下载了3个文件, 但其实只建立了', obj_pool.size(), '个对象')
输出结果(这里为了方便演示直接使用了sleep
方法,没有再用多线程模拟):
+ 从 https://download1.com 开始下载 文件1 - 文件1 下载完成 + 从 https://download2.com 开始下载 文件2 - 文件2 下载完成 + 从 https://download3.com 开始下载 文件3 - 文件3 下载完成 ******************** 下载了3个文件, 但其实只建立了 1 个对象
// 对象池 class ObjectPool { constructor() { this._pool = []; // } // 建立对象 create(Obj) { return this._pool.length === 0 ? new Obj(this) // 对象池中没有空闲对象,则建立一个新的对象 : this._pool.shift(); // 对象池中有空闲对象,直接取出,无需再次建立 } // 对象回收 recover(obj) { return this._pool.push(obj); } // 对象池大小 size() { return this._pool.length; } } // 模拟文件对象 class File { constructor(pool) { this.pool = pool; } // 模拟下载操做 download() { console.log(`+ 从 ${this.src} 开始下载 ${this.name}`); setTimeout(() => { console.log(`- ${this.name} 下载完毕`); // 下载完毕后, 将对象从新放入对象池 this.pool.recover(this); }, 100); } } /****************** 如下是测试函数 **********************/ let objPool = new ObjectPool(); let file1 = objPool.create(File); file1.name = "文件1"; file1.src = "https://download1.com"; file1.download(); let file2 = objPool.create(File); file2.name = "文件2"; file2.src = "https://download2.com"; file2.download(); setTimeout(() => { let file3 = objPool.create(File); file3.name = "文件3"; file3.src = "https://download3.com"; file3.download(); }, 200); setTimeout( () => console.log( `${"*".repeat(50)}\n下载了3个文件,但其实只建立了${objPool.size()}个对象` ), 1000 );
输出结果以下:
+ 从 https://download1.com 开始下载 文件1 + 从 https://download2.com 开始下载 文件2 - 文件1 下载完毕 - 文件2 下载完毕 + 从 https://download3.com 开始下载 文件3 - 文件3 下载完毕 ************************************************** 下载了3个文件,但其实只建立了2个对象