PHP前端开发

如何在 Python 中选择正确的设计模式(附示例)

百变鹏仔 3天前 #Python
文章标签 示例

设计模式是软件开发中常见问题的经过验证的解决方案。它们为解决设计问题提供了可重用的模板,从而提高了代码的可维护性和灵活性。

但是有这么多可用的设计模式,您如何知道针对给定问题在 python 中实现哪一种?在本文中,我们将探讨选择正确设计模式的步骤,并提供每个示例,以帮助您理解和有效应用它们。

1. 理解问题

选择设计模式的第一步是清楚地了解您要解决的问题。问自己以下问题:

预期的行为是什么?
系统有哪些限制?
可能的延伸或变化点有哪些?

2.设计模式分类

设计模式一般分为三类:

立即学习“Python免费学习笔记(深入)”;

创造:关注对象的创建。
结构:关注对象的组成。
行为:关注对象之间的交互。
确定与您的问题匹配的类别有助于缩小相关模式的数量。

3.选择合适的设计模式

了解问题及其类别后,查看该类别中的设计模式以找到最适合您情况的设计模式。考虑以下几点:

灵活性:该模式是否提供必要的灵活性?
复杂性:这不会带来不必要的复杂性吗?
可扩展性:它是否使未来的扩展更容易?

  1. python 中的设计模式示例辛格尔顿什么时候使用它?当您需要确保某个类只有一个实例并提供对该实例的全局访问点时。

python 示例:
`class singletonmeta(类型):
_instance = {}

def __call__(cls, *args, **kwargs):    if cls not in cls._instance:        cls._instance[cls] = super().__call__(*args, **kwargs)    return cls._instance[cls]

类 logger(metaclass=singletonmeta):
def log(自我, 消息):
print(f"[log]: {消息}")

使用

logger1 = logger()
logger2 = logger()

print(logger1 is logger2) # 输出:true

logger1.log(“运行中的单例模式。”)
`
为什么它有效?
singletonmeta 是一个控制 logger 实例创建的元类。如果实例已存在,则返回该实例,确保只有一个实例。

工厂
什么时候使用它?
当您有一个包含多个子类的父类,并且根据输入数据,您需要返回其中一个子类。

python 示例:
`类形状:
def 绘制(自身):
通过

类圆形(形状):
def 绘制(自身):
print("画一个圆。")

类正方形(形状):
def 绘制(自身):
print("画一个正方形。")

def shape_factory(shape_type):
如果 shape_type == "圆":
返回圆()
elif shape_type == "方形":
返回 square()
其他:
raise valueerror("未知的形状类型。")

使用

shape = shape_factory("圆")
shape.draw() # 输出:画一个圆。
`
为什么它有效?
工厂封装了对象创建逻辑,允许在不暴露底层逻辑的情况下创建实例。

观察
什么时候使用它?
当你有一个对象(主体)需要在状态发生变化时通知多个其他对象(观察者)。

python 示例:
`课程主题:
def init(self):
self._observers = []

def attach(self, observer):    self._observers.append(observer)def notify(self, message):    for observer in self._observers:        observer.update(message)

类观察者:
def 更新(自我,消息):
通过

类 emailobserver(观察者):
def 更新(自我,消息):
print(f"邮件通知:{message}")

类 smsobserver(观察者):
def 更新(自我,消息):
print(f"短信通知:{message}")

使用

主题=主题()
subject.attach(emailobserver())
subject.attach(smsobserver())

subject.notify("观察者模式已实现。")
`
为什么它有效?
主体维护观察者列表并通知他们变化,从而允许解耦通信。
策略
什么时候使用它?
当您有多种算法来执行一项任务并且您想要动态地交换它们时。

python 示例:
`导入类型

类文本处理器:
def init(自身,格式化程序):
self.formatter = types.methodtype(formatter, self)

def process(self, text):    return self.formatter(text)

def uppercase_formatter(self, text):
return text.upper()

def lowercase_formatter(self, text):
return text.lower()

使用

处理器 = textprocessor(uppercase_formatter)
print(processor.process("hello world")) # 输出:hello world

processor.formatter = types.methodtype(lowercase_formatter, 处理器)
print(processor.process("hello world")) # 输出:hello world
`
为什么它有效?
策略模式允许您通过为格式分配新函数来动态更改对象使用的算法。

装饰器
什么时候使用它?
当您想要动态地向对象添加新功能而不更改其结构时。

python 示例:
`def bold_decorator(func):
def 包装器():
返回“”+ func() +“
返回包装

def italic_decorator(func):
def 包装器():
返回 "" + func() + ""
返回包装

@bold_decorator
@italic_decorator
def say_hello():
返回“你好”

使用

print(say_hello()) # 输出:你好
`

它为什么有效?
装饰器允许您包装函数以添加功能,例如此处的格式化,而无需修改原始函数。

适应
什么时候使用它?
当您需要使用现有的类但其接口不符合您的需求时。

python 示例:
`europeansocketinterface 类:
定义电压(自身):通过
def live(self): 通过
def 中立(自我):通过

类europeansocket(europeansocketinterface):
定义电压(自身):
返回230

def live(self):    return 1def neutral(self):    return -1

usasocketinterface 类:
定义电压(自身):通过
def live(self): 通过
def 中立(自我):通过

类适配器(usasocketinterface):
def init(self, european_socket):
self.european_socket = european_socket

def voltage(self):    return 110def live(self):    return self.european_socket.live()def neutral(self):    return self.european_socket.neutral()

使用

euro_socket = europeansocket()
适配器 = 适配器(euro_socket)
print(f"电压:{adapter.voltage()}v") # 输出:电压:110v
`
适配器将一个类的接口转换为客户端期望的另一个接口,从而允许不兼容的接口之间兼容。

命令
什么时候使用它?
当您想将请求封装为对象时,允许您使用不同的请求、队列或日志记录来配置客户端。

python 示例:
`类命令:
def 执行(自我):
通过

类 lightoncommand(命令):
def init(self, light):
self.light = 光

def execute(self):    self.light.turn_on()

类 lightoffcommand(命令):
def init(self, light):
self.light = 光

def execute(self):    self.light.turn_off()

类光:
def open_on(自身):
print("灯亮了")

def turn_off(self):    print("Light turned OFF")

远程控制类:
def 提交(自我,命令):
命令.execute()

使用

光 = light()
on_command = lightoncommand(灯光)
off_command = lightoffcommand(光)

远程 = remotecontrol()
remote.submit(on_command) # 输出:灯亮
remote.submit(off_command) # 输出:灯关闭
`
为什么它有效?
命令模式将操作转换为对象,允许配置、排队或取消操作。

5. 结论

在 python 中选择正确的设计模式需要清楚地了解要解决的问题和可用的模式。通过对问题进行分类并分析每种模式的优点,您可以选择提供最有效解决方案的一种。

请记住,设计模式是改进代码的工具,而不是需要遵循的严格规则。明智地使用它们来编写干净、可维护且可扩展的 python 代码。

6. 额外资源

书籍:
设计模式:可重用面向对象软件的元素 erich gamma 等人
埃里克·弗里曼 (eric freeman) 和伊丽莎白·罗布森 (elisabeth robson) 的 head first 设计模式。
网站:
重构大师
深入研究设计模式
感谢您的阅读!欢迎在评论中分享您使用 python 设计模式的经验。