定義一系列算法,把它們一一封裝起來,而且使它們能夠相互替
換。本模式使得算法能夠獨立於使用它的客戶而變化。python
假如一個網店制定了下述折扣規則。算法
簡單起見,咱們假定一個訂單一次只能享用一個折扣。
from abc import ABC, abstractclassmethod from collections import namedtuple Customer = namedtuple('Customer', 'name fidelity') # 相似於購物車 class LineItem(object): def __init__(self, product, quantity, price): self.product = product self.quantity = quantity self.price = price def total(self): return self.price * self.quantity # 上下文是 Order class Order(object): def __init__(self, customer, cart, promotion=None): self.customer = customer self.cart = list(cart) self.promotion = promotion def total(self): ###4 if not hasattr(self, '__total'): self.__total = sum(item.total() for item in self.cart) return self.__total def due(self): ###5 if self.promotion is None: discount = 0 else: discount = self.promotion.discount(self) ###6 return self.total() - discount def __repr__(self): ###2 fmt = '<Order total: {:.2f} due: {:.2f}>' return fmt.format(self.total(), self.due()) ###3 # 策略:抽象基類 class Promotion(ABC): @abstractclassmethod def discount(self, order): """返回折扣金額(正值)""" class FidelityPromo(Promotion): # 第一個具體策略 """爲積分爲1000或以上的顧客提供5%折扣""" def discount(self, order): ###7 return order.total() * .05 if order.customer.fidelity >= 1000 else 0 class BulkItemPromo(Promotion): # 第二個具體策略 """單個商品爲20個或以上時提供10%折扣""" def discount(self, order): discount = 0 for item in order.cart: if item.quantity >= 20: discount += item.total() * .1 return discount class LargeOrderPromo(Promotion): # 第三個具體策略 """訂單中的不一樣商品達到10個或以上時提供7%折扣""" def discount(self, order): distinct_items = {item.product for item in order.cart} if len(distinct_items) >= 10: return order.total() * .07 return 0 if __name__ == "__main__": # 兩位顧客 一個積分是0 一個是1100 longe = Customer('longe', 0) liang = Customer('Ann Smith', 1100) # 購物的商品 cart = [LineItem('banana', 4, .5), LineItem('apple', 10, 1.5), LineItem('watermellon', 5, 5.0)] # 這樣去調用 print(Order(longe, cart, FidelityPromo())) ###111 print(Order(liang, cart, FidelityPromo())) banana_cart = [LineItem('banana', 30, .5), LineItem('apple', 10, 1.5)] print(Order(longe, banana_cart, BulkItemPromo())) long_order = [LineItem(str(item_code), 1, 1.0) for item_code in range(10)] print(Order(longe, long_order, LargeOrderPromo()))
from collections import namedtuple Customer = namedtuple('Customer', 'name fidelity') class LineItem: def __init__(self, product, quantity, price): self.product = product self.quantity = quantity self.price = price def total(self): return self.price * self.quantity class Order: # 上下文 def __init__(self, customer, cart, promotion=None): self.customer = customer self.cart = list(cart) self.promotion = promotion def total(self): if not hasattr(self, '__total'): self.__total = sum(item.total() for item in self.cart) return self.__total def due(self): if self.promotion is None: discount = 0 else: discount = self.promotion(self) return self.total() - discount def __repr__(self): fmt = '<Order total: {:.2f} due: {:.2f}>' return fmt.format(self.total(), self.due()) def fidelity_promo(order): """爲積分爲1000或以上的顧客提供5%折扣""" return order.total() * .05 if order.customer.fidelity >= 1000 else 0 def bulk_item_promo(order): """單個商品爲20個或以上時提供10%折扣""" discount = 0 for item in order.cart: if item.quantity >= 20: discount += item.total() * .1 return discount def large_order_promo(order): """訂單中的不一樣商品達到10個或以上時提供7%折扣""" distinct_items = {item.product for item in order.cart} if len(distinct_items) >= 10: return order.total() * .07 return 0 joe = Customer('John Doe', 0) ann = Customer('Ann Smith', 1100) cart = [LineItem('banana', 4, .5),LineItem('apple', 10, 1.5),LineItem('watermellon', 5, 5.0)] print(Order(joe, cart, fidelity_promo)) print(Order(ann, cart, fidelity_promo)) banana_cart = [LineItem('banana', 30, .5),LineItem('apple', 10, 1.5)] print(Order(joe, banana_cart, bulk_item_promo)) long_order = [LineItem(str(item_code), 1, 1.0) for item_code in range(10)] print(Order(joe, long_order, large_order_promo)) print(Order(joe, cart, large_order_promo))
from collections import namedtuple Customer = namedtuple('Customer', 'name fidelity') class LineItem: def __init__(self, product, quantity, price): self.product = product self.quantity = quantity self.price = price def total(self): return self.price * self.quantity class Order: # 上下文 def __init__(self, customer, cart, promotion=None): self.customer = customer self.cart = list(cart) self.promotion = promotion def total(self): if not hasattr(self, '__total'): self.__total = sum(item.total() for item in self.cart) return self.__total def due(self): if self.promotion is None: discount = 0 else: discount = self.promotion(self) return self.total() - discount def __repr__(self): fmt = '<Order total: {:.2f} due: {:.2f}>' return fmt.format(self.total(), self.due()) def fidelity_promo(order): """爲積分爲1000或以上的顧客提供5%折扣""" return order.total() * .05 if order.customer.fidelity >= 1000 else 0 def bulk_item_promo(order): """單個商品爲20個或以上時提供10%折扣""" discount = 0 for item in order.cart: if item.quantity >= 20: discount += item.total() * .1 return discount def large_order_promo(order): """訂單中的不一樣商品達到10個或以上時提供7%折扣""" distinct_items = {item.product for item in order.cart} if len(distinct_items) >= 10: return order.total() * .07 return 0 joe = Customer('John Doe', 0) ann = Customer('Ann Smith', 1100) cart = [LineItem('banana', 4, .5), LineItem('apple', 10, 1.5), LineItem('watermellon', 5, 5.0)] banana_cart = [LineItem('banana', 30, .5), LineItem('apple', 10, 1.5)] long_order = [LineItem(str(item_code), 1, 1.0) for item_code in range(10)] ## 找出最大的折扣 promos = [fidelity_promo, bulk_item_promo, large_order_promo] def best_promo(order): """選擇可用的最佳折扣""" # print([promo(order) for promo in promos]) return max(promo(order) for promo in promos) print(Order(joe, long_order, best_promo)) print(Order(joe, banana_cart, best_promo)) print(Order(ann, cart, best_promo) )
promos = [globals()[name] for name in globals() if name.endswith('_promo') and name != 'best_promo'] print(promos)
另外一個可行的方法是將全部的策略函數都存放在一個單獨的模塊中
除了 best_promo,這裏咱們將 3 個策略函數存放在了 promotions.py 中設計模式
下面的代碼中,最大的變化時內省名爲 promotions 的獨立模塊,構建策略函數列表。
注意,下面要導入 promotions 模塊,以及高階內省函數的 inspect 模塊app
import inspect import promotions promos = [func for name, func in inspect.getmembers(promotions, inspect.isfunction)] def best_promo(order): """選擇可用的最佳折扣 """ return max(promo(order) for promo in promos) print(Order(joe, long_order, best_promo)) print(Order(joe, banana_cart, best_promo)) print(Order(ann, cart, best_promo) )
class MacroCommand: """一個執行一組命令的命令""" def __init__(self, commands): self.commands = list(commands) def __call__(self): for command in self.commands: command()
1.python 對某些設計默認 能夠用純函數來實現, 不用能夠去寫類
2.設計模式得看看了ide