行为型模式
责任链模式
内容:使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。
将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
角色:
抽象处理者(Handler)
具体处理者(ConcreteHandler)
客户端(Client)
from abc import ABCMeta, abstractmethod class Handler(metaclass=ABCMeta):
@abstractmethod
def handle_leave(self, day):
pass class GeneralManager(Handler):
def handle_leave(self, day):
if day <= 10:
print("总经理准假%d天" % day)
else:
print("你还是辞职吧") class DepartmentManager(Handler):
def __init__(self):
self.next = GeneralManager() def handle_leave(self, day):
if day <= 5:
print("部门经理准假%s天" % day)
else:
print("部门经理职权不足")
self.next.handle_leave(day) class ProjectDirector(Handler):
def __init__(self):
self.next = DepartmentManager() def handle_leave(self, day):
if day <= 3:
print("项目主管准假%d天" % day)
else:
print("项目主管职权不足")
self.next.handle_leave(day) day = 12
h = ProjectDirector()
h.handle_leave(day)
适用场景:
有多个对象可以处理一个请求,哪个对象处理由运行时决定
在不明确接收者的情况下,向多个对象中的一个提交一个请求
优点:
降低耦合度:一个对象无需知道是其他哪一个对象处理其请求
观察者模式
内容:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时, 所有依赖于它的对象都得到
通知并被自动更新。观察者模式又称“发布-订阅”模式
角色:
抽象主题(Subject)
具体主题(ConcreteSubject)——发布者
抽象观察者(Observer)
具体观察者(ConcreteObserver)——订阅者
from abc import ABCMeta, abstractmethod class Observer(metaclass=ABCMeta): # 抽象订阅者
@abstractmethod
def update(self, notice): # notice 是一个Notice类的对象,以便过去Notice更新的消息
pass class Notice: # 抽象发布者
def __init__(self):
self.observers = [] def attach(self, obs):
self.observers.append(obs) # 添加订阅 def detach(self, obs):
self.observers.remove(obs) # 取消订阅 def notify(self): # 向订阅者推送消息
for obs in self.observers:
obs.update(self) # 对订阅者的消息进行更新 class StaffNotice(Notice): # 具体发布者
def __init__(self, company_info=None):
super().__init__()
self.__company_info = company_info @property
def company_info(self):
return self.__company_info @company_info.setter
def company_info(self, info):
self.__company_info = info
self.notify() # 一有改变就向订阅者发送消息,并调用订阅者的消息更新方法 class Staff(Observer):
def __init__(self):
self.company_info = None def update(self, notice): # 获取消息更新的方法
self.company_info = notice.company_info # Client notice = StaffNotice("初始公司信息") s1 = Staff()
s2 = Staff() notice.attach(s1)
notice.attach(s2) notice.company_info = "公司今年业绩非常好,给大家发奖金!!!"
print(s1.company_info)
print(s2.company_info)
notice.detach(s2)
notice.company_info = "公司明天放假!!!"
print(s1.company_info)
print(s2.company_info) """
公司今年业绩非常好,给大家发奖金!!!
公司今年业绩非常好,给大家发奖金!!!
公司明天放假!!!
公司今年业绩非常好,给大家发奖金!!! """
适用场景:
当一个抽象模型有两方面,其中一个方面依赖于另一个方面。将这两者封装在独立对象中以使它们可以各自独立地改变和
复用。
当对一个对象的改变需要同时改变其它对象,而不知道具体有多少对象有待改变。
当一个对象必须通知其它对象,而它又不能假定其它对象是谁。换言之,你不希望这些对象是紧密耦合的。
优点:
目标和观察者之间的抽象耦合最小
支持广播通信
策略模式
内容:定义一系列的算法,把它们一个个封装起来,并且使它们可相互替换。本模式使
得算法可独立于使用它的客户而变化。
角色:
抽象策略(Strategy)
具体策略(ConcreteStrategy)
上下文(Context)
from abc import ABCMeta, abstractmethod class Strategy(metaclass=ABCMeta):
@abstractmethod
def execute(self, data):
pass class FastStrategy(Strategy):
def execute(self, data):
print("用较快的策略处理%s" % data) class SlowStrategy(Strategy):
def execute(self, data):
print("用较慢的策略处理%s" % data) class Context:
def __init__(self, strategy, data): # 将对数据处理的方法封装起来
self.data = data
self.strategy = strategy def set_strategy(self, strategy): #设置策略
self.strategy = strategy def do_strategy(self): # 执行策略的方法
self.strategy.execute(self.data) data = "[...]"
s1 = FastStrategy()
s2 = SlowStrategy()
context = Context(s1, data)
context.do_strategy()
context.set_strategy(s2)
context.do_strategy()
优点:
定义了一系列可重用的算法和行为
消除了一些条件语句
可以提供相同行为的不同实现
缺点:
客户必须了解不同的策略
模板方法模式
内容:定义一个操作中的算法的骨架,而将一些步骤延迟到子类中。模板方法
使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。
角色:
抽象类(AbstractClass):定义抽象的原子操作(钩子操作);实现一个模
板方法作为算法的骨架。
具体类(ConcreteClass):实现原子操作
from abc import ABCMeta, abstractmethod
from time import sleep class Window(metaclass=ABCMeta):
@abstractmethod
def start(self):
pass @abstractmethod
def repaint(self):
pass @abstractmethod
def stop(self): # 原子操作/钩子操作
pass def run(self): # 模板方法
self.start()
while True:
try:
self.repaint()
sleep(1)
except KeyboardInterrupt:
break
self.stop() class MyWindow(Window):
def __init__(self, msg):
self.msg = msg def start(self):
print("窗口开始运行") def stop(self):
print("窗口结束运行") def repaint(self):
print(self.msg) MyWindow("Hello...").run()
适用场景:
一次性实现一个算法的不变的部分
各个子类中的公共行为应该被提取出来并集中到一个公共父类中以
避免代码重复
控制子类扩展