zoukankan      html  css  js  c++  java
  • sjms-4 行为型模式

    行为型模式

    责任链模式


    内容:使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。
    将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
    角色:
    抽象处理者(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()
    

      


    适用场景:
    一次性实现一个算法的不变的部分
    各个子类中的公共行为应该被提取出来并集中到一个公共父类中以
    避免代码重复
    控制子类扩展

  • 相关阅读:
    图文详解k8s自动化持续集成之GitLab CI/CD
    GitLab CI/CD 进行持续集成
    高性能伪事务之Lua in Redis
    面向对象之组合VS继承:继承过时了?
    Go依赖模块版本之Module避坑使用详解
    Maven3路程(三)用Maven创建第一个web项目(1)
    C#通过WIN32 API实现嵌入程序窗体
    Selenium自動化測試(Python+VS2013)-基礎篇-環境安裝
    VS2013中Python学习笔记[环境搭建]
    win7 windows server 2008R2下 https SSL证书安装的搭配(搭配https ssl本地测试环境)
  • 原文地址:https://www.cnblogs.com/wenyule/p/10534240.html
Copyright © 2011-2022 走看看