python 设计模式

1.创建型模式

  • 单例模式
    • 内容:单例模式,可以确保某个类只有一个实例存在。
    • 角色:
      • 单例(singleton)
    • demon:
from threading import Thread, Lock


class Singleton:
   """线程安全的单例模式"""
   _instance_lock = Lock()
   _init_lock = Lock()

   def __new__(cls, *args, **kwargs):
       # 确保只有一个实例
       with Singleton._instance_lock:
           if not hasattr(Singleton, '_instance'):
               print('实例第一次被创建')
               Singleton._instance = object.__new__(cls)
       return Singleton._instance

   def __init__(self):
       # 只运行一次init
       with Singleton._init_lock:
           if not hasattr(Singleton, '_first_init'):
               print('实例第一次被初始化')
               Singleton._first_init = True # 设置类属性flag


def task(i, threadPool):
   if isinstance(threadPool, list):
       obj = Singleton()
       if obj not in  threadPool:
           threadPool.append(obj)
       


if __name__ == "__main__":
   threadPool = []
   for i in range(10):
       t = Thread(target=task, args=(i, threadPool))
       t.start()
   print(threadPool) # 长度为1相对于循环10次只创建一个实例
  • 简单工厂模式
    • 内容:不直接向客户端暴露对象创建的实现细节,通过一个工厂来负责创建产品类实例。
    • 角色:
      • 工厂角色(Creator)
      • 抽象产品角色(Product)
      • 具体产品角色(Concrete Product)
    • demon:
from abc import ABCMeta, abstractmethod

class Payment(metaclass = ABCMeta):
    """抽象产品角色"""
    @abstractmethod
    def pay(self, money):
        pass

class Alipay(Payment):
   """具体产品角色"""
    def pay(self, money):
        print(f"支付宝支付{money}元")

class WechatPay(Payment):
   """具体产品角色"""
    def pay(self, money):
        print(f"微信支付{money}元")

class PaymentFactory:
   """工厂角色"""
    def create_payment(self, method):
        if method == 'alipay':
            return Alipay()
        elif method == 'wechat':
            return WechatPay()
        else:
            raise TypeError('tmd 找不到')

pf = PaymentFactory() # 工厂类实例
p = pf.create_payment('alipay') # 根据需求类型生产对应类实例,确保同一工厂多个产品
p.pay(888)
  • 工厂方法模式
    • 内容:在简单工厂的基础上,抽象出不同的工厂,每个工厂对应生产自己的产品,这就是工厂方法。
    • 角色:
      • 抽象工厂角色(Creator)
      • 具体工厂角色(Concrete Creator)
      • 抽象产品角色(Product)
      • 具体产品角色(Concrete Product)
    • demon:
from abc import ABCMeta, abstractmethod

class Payment(metaclass = ABCMeta):
    """抽象产品角色"""
    @abstractmethod
    def pay(self, money):
        pass

class Alipay(Payment):
    """具体产品角色"""
    def pay(self, money):
        print(f"支付宝支付{money}元")

class WechatPay(Payment):
    """具体产品角色"""
    def pay(self, money):
        print(f"微信支付{money}元")

class PaymentFactory(metaclass = ABCMeta):
    """抽象工厂角色"""
    def create_payment(self):
        pass
    
class AlipayFactory(PaymentFactory):
    """具体工厂角色"""
    def create_payment(self):
        return Alipay()

class WechatFactory(PaymentFactory):
    """具体工厂角色"""
    def create_payment(self):
        return WechatPay()

pf = AlipayFactory() # 工厂方法类实例
p = pf.create_payment() # 根据需求类型调用对应方法工厂生产产品
p.pay(888)
  • 抽象工厂模式
    • 内容:定义一个工厂类接口,让工厂子类创建一系列相关或依赖的对象。
    • 角色:
      • 抽象工厂角色(Creator)
      • 具体工厂角色(Concrete Creator)
      • 抽象产品角色(Product)
      • 具体产品角色(Concrete Product)
    • demon:
from abc import ABCMeta, abstractmethod

class PhoneShell(metaclass = ABCMeta):
    """抽象产品角色"""
    @abstractmethod
    def show_shell(self):
        pass

class CPU(metaclass = ABCMeta):
    """抽象产品角色"""
    @abstractmethod
    def show_cpu(self):
        pass

class OS(metaclass = ABCMeta):
    """抽象产品角色"""
    @abstractmethod
    def show_os(self):
        pass

class PhoneFactory(metaclass = ABCMeta):
    """抽象工厂角色"""
    @abstractmethod
    def make_shell(self):
        pass

    @abstractmethod
    def make_cpu(self):
        pass

    @abstractmethod
    def make_os(self):
        pass

class SmallShell(PhoneShell):
    """具体产品角色"""
    def show_shell(self):
        return Alipay()


class SmallShell(PhoneShell):
    """具体产品角色"""
    def show_shell(self):
        print("普通小手机壳")

class BigShell(PhoneShell):
    """具体产品角色"""
    def show_shell(self):
        print("普通大手机壳")

class AppLeShell(PhoneShell):
    """具体产品角色"""
    def show_shell(self):
        print("苹果手机壳")

class SnapDrageon(CPU):
    """具体产品角色"""
    def show_cpu(self):
        print("晓龙cpu")

class MediaTekCPU(CPU):
    """具体产品角色"""
    def show_cpu(self):
        print("联发科cpu")

class AppleCPU(CPU):
    """具体产品角色"""
    def show_cpu(self):
        print("苹果cpu")

class IOS(OS):
    """具体产品角色"""
    def show_os(self):
        print("苹果os")

class Android(OS):
    """具体产品角色"""
    def show_os(self):
        print("Android os")

class MiFactory(PhoneFactory):
    """具体工厂角色"""
    def make_cpu(self):
        return SnapDrageon()
    
    def make_os(self):
        return Android()
    
    def make_shell(self):
        return BigShell()

class HuaweiFactory(PhoneFactory):
    """具体工厂角色"""
    def make_cpu(self):
        return MediaTekCPU()
    
    def make_os(self):
        return Android()
    
    def make_shell(self):
        return SmallShell()

class AppleFactory(PhoneFactory):
    """具体工厂角色"""
    def make_cpu(self):
        return AppleCPU()
    
    def make_os(self):
        return OS()
    
    def make_shell(self):
        return AppLeShell()

class Phone:
    def __init__(self, cpu, os, shell):
        self.cpu = cpu
        self.os = os
        self.shell = shell
    
    def show_info(self):
        print('手机信息')
        self.cpu.show_cpu()
        self.os.show_os()
        self.shell.show_shell()

def make_phone(factory):
    cpu = factory.make_cpu()
    os = factory.make_os()
    shell = factory.make_shell()
    return Phone(cpu, os, shell)

pf = make_phone(MiFactory())
pf.show_info()
  • 建造者模式:
    • 内容:将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示。
    • 角色:
      • 抽象建造者(Builder)
      • 具体建造者(Concrete Builder)
      • 指挥者(Director)
      • 产品(Product)
    • demon:
from abc import ABCMeta, abstractmethod

class Player:
    def __init__(self, face = None, body = None, arm = None, leg = None):
        self.face = face
        self.body = body
        self.arm = arm
        self.leg = leg
    
    def __str__(self):
        return f"{self.face} - {self.body} - {self.arm} - {self.leg}"

class PlayerBuilder(metaclass = ABCMeta):
    """ """
    @abstractmethod
    def build_face(self):
        pass

    @abstractmethod
    def build_body(self):
        pass

    @abstractmethod
    def build_arm(self):
        pass

    @abstractmethod
    def build_leg(self):
        pass

class SexGirlBuilder(PlayerBuilder):
    def __init__(self):
        self.player = Player()
    
    def build_face(self):
        self.player.face = "漂亮脸蛋"
    
    def build_body(self):
        self.player.body = "苗条"
    
    def build_arm(self):
        self.player.arm = "漂亮胳膊"
    
    def build_leg(self):
        self.player.leg = "大长腿"

class Monster(Player):
    def __init__(self):
        self.player = Player()
    
    def build_face(self):
        self.player.face = "怪兽脸"
    
    def build_body(self):
        self.player.body = "胖"
    
    def build_arm(self):
        self.player.arm = "长毛的胳膊"
    
    def build_leg(self):
        self.player.leg = "大长腿"

class PlayerDirector:
    def build_player(self, builder):
        builder.build_body()
        builder.build_face()
        builder.build_leg()
        builder.build_arm()
        return builder.player

builder = SexGirlBuilder()
director = PlayerDirector()
p = director.build_player(builder)
print(p)

2.结构型模式

  • 适配器模式
    • 内容:将一个类的接口转换成客户希望的另一个接口。适配器模式使得原本由于接口不兼容而不能在一起工作的那些类可以一起工作。
    • 实现方式:
      • 类适配器:使用多继承
      • 对象适配器:使用组合
    • 角色:
      • 目标接口(Target)
      • 待适配的类(Adaptee)
      • 适配器(Adapter)
    • demon:
from abc import ABCMeta, abstractmethod

class Payment(metaclass = ABCMeta):
    @abstractmethod
    def pay(self, money):
        pass

class Alipay(Payment):
    def pay(self, money):
        """目标接口"""
        print(f"支付宝支付{money}元")

class WechatPay(Payment):
    def pay(self, money):
        """目标接口"""
        print(f"微信支付{money}元")

class BanPay:
    """待适配的类"""
    def cost(self, money):
       print(f'银联支付{money}')

class ApplePay:
    def cost(self, money):
        print(f'苹果支付{money}')

# class NewBankPay(Payment, BankPay):
#     """类适配器类"""
#     def pay(self, money):
#         self.cost(money)

# p = NewBankPay()
# p.pay(110)

class PaymentAdapter(Payment):
    """对象适配器"""
    def __init__(self, payment):
        self.payment = Payment
    
    def pay(self, money):
        self.payment.cost(money)

p = PaymentAdapter(ApplePay())
p.pay(100)
  • 桥模式
    • 内容:将一个事物的两个维度分离,使其都可以独立地变化。
    • 角色:
      • 抽象(Abstraction)
      • 细化抽象(RefinedAbstraction)
      • 实现者(Implementor)
      • 具体实现者(ConcreteImplementor)
    • demon:
from abc import ABCMeta, abstractmethod

class Shape(metaclass=ABCMeta):
    def __init__(self, color):
        self.color = color

    @abstractmethod
    def draw(self):
        pass

class Color(metaclass=ABCMeta):
    @abstractmethod
    def paint(self, shape):
        pass

class Rectangle(Shape):
    """抽象"""
    name = "长方形"
    def draw(self):
        # 长方形绘制逻辑
        self.color.paint(self):

class Circle(Shape):
    """抽象"""
    name = "圆形"
    def draw(self):
        # 圆形绘制逻辑
        self.color.paint(self)

class Red(Color):
    """实现"""
    def paint(self, shape):
        peint(f"红色的{shape.name}")

class Green(Color):
    """实现"""
    def paint(self, shape):
        peint(f"绿色的{shape.name}")

shape = Rectangle(Red())
shape.draw()
  • 组合模式
    • 内容:将对象组合合成树形结构以表示“部分-整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。
    • 角色:
      • 抽象组件(Component)
      • 叶子组件(Leaf)
      • 复合组件(Composite)
      • 客户端(Client)
    • demon:
from abc import ABCMeta, abstractmethod

class Graphic(metaclass=ABCMeta):
    """抽象组件"""
    @abstractmethod
    def draw(self):
        pass

def Point(Graphic):
    """叶子组件"""
    def __init__(self, x, y):
        self.x = x
        self.y = y

    def __str__(self):
        return "点({self.x},{self.y})"
    
    def draw(self):
        print(str(self))

def Line:
    """叶子组件"""
    def __init__(self, p1, p2):
        self.p1 = p1
        self.p2 = p2

    def __str__(self):
        return "线段[{self.p1},{self.p2}]"
    
    def draw(self):
        print(str(self))

class Picture:
    """复合组件"""
    def __init__(self, iterable):
        self.children = []
        for g in iterable:
            self.add(g)
    
    def add(self, graphic):
        self.children.append(graphic)
    
    def draw(self):
        print("----------复合图形-----------")
        for g in self.children:
            g.draw()
         print("----------复合图形-----------")

# 客户端
p1 = Point(2, 3)
l1 = Line(Point(3, 4), Ponit(6, 7))
l2 = Line(Point(1, 5), Ponit(2, 8))
pic1 = Picture([p1, l1, l2])
pic1.draw()

p2 = Point(4, 4)
l3 = Line(Point(1, 1), Point(0, 0))
pic2 = Picture([pic1, pic2])

pic = Picture([pic1, pic2])
pic.draw()
  • 外观模式
    • 内容:为子系统中的一组接口提供一个一致的界面,外观模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。
    • 角色:
      • 外观(facade)
      • 子系统(subsystem classes)
    • demon:
class CPU:
    """子系统"""
    def run(self):
        print("CPU开始运行")
    
    def stop(self):
        print("CPU停止运行")

class Disk:
    """子系统"""
    def run(self):
        print("键盘开始运行")
    
    def stop(self):
        print("键盘停止运行")

class Memory:
    """子系统"""
    def run(self):
        print("内存通电")
    
    def stop(self):
        print("内存断电")

class Computer:
    """外观"""
    def __init__(self):
        self.cpu = CPU()
        self.disk = Disk()
        self.memory = Memory()

    def run(self):
        self.cpu.run()
        self.disk.run()
        self.memory.run()
    
    def stop(self):
        self.cpu.stop()
        self.disk.stop()
        self.memory.stop()

computer = Computer()
computer.run()
computer.stop()

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,204评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,091评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,548评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,657评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,689评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,554评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,302评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,216评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,661评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,851评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,977评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,697评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,306评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,898评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,019评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,138评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,927评论 2 355

推荐阅读更多精彩内容

  • 在面向对象编程中,工厂表示一个负责创建其他类型的对象的类。通常作为工厂的类会实现多个关联的方法,客户端通过某些参数...
    rollingstarky阅读 850评论 0 0
  • 工作时间一长,需求多而杂,往往难得有时间去仔细思量代码如何写的更加优雅,习惯使然。设计模式本身不是一个神秘的东西,...
    __七把刀__阅读 1,281评论 0 5
  • 1.设计模式 设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用...
    华丽的微笑阅读 518评论 0 0
  • 1设计模式 设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。...
    五行缺觉阅读 297评论 0 1
  • 抽象工厂 抽象工厂设计模式是抽象方法的一种泛化。概括来说,一个抽象工厂是(逻辑上的)一组工厂方法,其中的每个工厂方...
    英武阅读 5,312评论 0 52