享元模式
(一)什么是享元模式
享元设计模式通过为相似对象引入数据共享来最小化内存使用,提升性能(请参考网页。
(二)使用场景
- 系统中存在大量的相似对象时,可以选择享元模式提高资源利用率。咖啡订购平台比较小,若假设一个电商平台,每个买家和卖家建立起买卖关系后,买家对象和卖家对象都是占用资源的。如果一个卖家同时与多个买家建立起买卖关系呢?此时享元模式的优势就体现出来了;
- 需要缓冲池的场景中,可以使用享元模式。如进程池,线程池等技术,就可以使用享元模式(事实上,很多的池技术中已经使得了享元模式)。
(三)优缺点
优点
- 减少重复对象,大大节约了系统资源。
缺点
- 享元模式虽然节约了系统资源,但同时也提高了系统的复杂性,尤其当遇到外部状态和内部状态混在一起时,需要先将其进行分离,才可以使用享元模式。否则,会引起逻辑混乱或业务风险。
- 享元模式中需要额外注意线程安全问题。
(四)代码实现
# 假设有一个网上咖啡选购平台,客户可以在该平台上下订单订购咖啡,平台会根据用户位置进行线下配送。
class Coffee:
name = ''
price =0
def __init__(self,name):
self.name = name
self.price = len(name)
# 在实际业务中,咖啡价格应该是由配置表进行配置,
# 或者调用接口获取等方式得到,此处为说明享元模式,
# 将咖啡价格定为名称长度,只是一种简化
def show(self):
print ("咖啡的名字:%s 价格:%s" % (self.name, self.price))
# 顾客类
class Customer:
name=""
def __init__(self,name):
self.name=name
def order(self,coffee_name):
print ("%s 点了一杯咖啡:%s" % (self.name, coffee_name))
return Coffee(coffee_name)
# 在咖啡实例化前,增加一个控制实例化的类:咖啡工厂。
# getCoffeeCount直接返回当前实例个数
class CoffeeFactory():
coffee_dict = {}
def getCoffee(self, name):
if name not in self.coffee_dict:
self.coffee_dict[name] = Coffee(name)
return self.coffee_dict[name]
def getCoffeeCount(self):
return len(self.coffee_dict)
# Customer类重写
class Customer2:
coffee_factory=""
name=""
def __init__(self,name,coffee_factory):
self.name=name
self.coffee_factory=coffee_factory
def order(self,coffee_name):
print ("%s 点了一杯咖啡:%s" % (self.name, coffee_name))
return self.coffee_factory.getCoffee(coffee_name)
# 短时间内有多人订了咖啡,业务模拟如下
if __name__=="__main__":
coffee_factory=CoffeeFactory()
customer_1=Customer2("A 客户",coffee_factory)
customer_2=Customer2("B 客户",coffee_factory)
customer_3=Customer2("C 客户",coffee_factory)
c1_capp=customer_1.order("cappuccino")
c1_capp.show()
c2_mocha=customer_2.order("mocha")
c2_mocha.show()
c3_capp=customer_3.order("mocha")
c3_capp.show()
print ("咖啡的实例个数:%s" % coffee_factory.getCoffeeCount ())