Python 在开发中的设计模式有哪些?怎样使用?

news2025/1/12 20:41:06

大家好!我是爱摸鱼的小鸿,关注我,收看每期的编程干货。

今天我们要聊点硬核的——设计模式。不过,不用担心,我会带着热情来跟你分享这些看似枯燥的知识点。让我们一起从“代码搬砖工”蜕变成“代码艺术家”吧!

目录

    • 一、设计模式是什么鬼?
    • 二、创建型模式
    • 三、结构型模式
    • 四、行为型模式
    • 五、结语
    • 六、作者Info

一、设计模式是什么鬼?

什么是设计模式呢?
设计模式(Design Pattern)是软件工程中,针对某一特定问题的最佳解决方案。简而言之,它是前辈们踩过无数坑后总结出的经验教训,帮助我们更优雅、更高效地写代码。

为什么需要设计模式?

  • 提高代码可读性:结构化的代码更容易理解。
  • 提升代码复用性:避免重复造轮子。
  • 增强系统扩展性:设计良好的模式使得系统更易扩展。
  • 降低维护成本:更少的bug和更简单的修改。


好了,废话不多说,接下来我们进入正题,一起来看看 Python 中的那些常用设计模式以及它们的使用方法吧!

二、创建型模式

单例模式(Singleton)
单例模式保证一个类只有一个实例,并提供一个全局访问点。这个模式在需要唯一对象的场景下非常有用,比如日志记录器、数据库连接池等。

代码示例:

class Singleton:
    _instance = None

    def __new__(cls, *args, **kwargs):
        if not cls._instance:
            cls._instance = super().__new__(cls, *args, **kwargs)
        return cls._instance

# 测试单例模式
singleton1 = Singleton()
singleton2 = Singleton()

print(singleton1 is singleton2)  # True



工厂模式(Factory Method)
工厂模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。这使得类的实例化延迟到子类。

代码示例:

from abc import ABC, abstractmethod

class Product(ABC):
    @abstractmethod
    def operation(self):
        pass

class ConcreteProductA(Product):
    def operation(self):
        return "Result of ConcreteProductA"

class ConcreteProductB(Product):
    def operation(self):
        return "Result of ConcreteProductB"

class Creator(ABC):
    @abstractmethod
    def factory_method(self):
        pass

    def some_operation(self):
        product = self.factory_method()
        return f"Creator: The same creator's code has just worked with {product.operation()}"

class ConcreteCreatorA(Creator):
    def factory_method(self):
        return ConcreteProductA()

class ConcreteCreatorB(Creator):
    def factory_method(self):
        return ConcreteProductB()

# 测试工厂模式
def client_code(creator: Creator):
    print(f"Client: I'm not aware of the creator's class, but it still works.\n{creator.some_operation()}")

client_code(ConcreteCreatorA())
client_code(ConcreteCreatorB())

三、结构型模式

适配器模式(Adapter)
适配器模式允许接口不兼容的类可以一起工作。它通过包装一个对象来转换它的接口,使得原本由于接口不兼容而不能一起工作的类可以一起工作。

代码示例:

class Target:
    def request(self):
        return "Target: The default target's behavior."

class Adaptee:
    def specific_request(self):
        return ".eetpadA eht fo roivaheb laicepS"

class Adapter(Target):
    def __init__(self, adaptee: Adaptee):
        self.adaptee = adaptee

    def request(self):
        return f"Adapter: (TRANSLATED) {self.adaptee.specific_request()[::-1]}"

# 测试适配器模式
adaptee = Adaptee()
print(f"Adaptee: {adaptee.specific_request()}")

adapter = Adapter(adaptee)
print(adapter.request())



装饰器模式(Decorator)
装饰器模式通过在运行时动态地将责任附加到对象上来扩展它的功能。装饰器提供了一个灵活的替代继承的方法来扩展功能。

代码示例:

class Component:
    def operation(self):
        pass

class ConcreteComponent(Component):
    def operation(self):
        return "ConcreteComponent"

class Decorator(Component):
    def __init__(self, component: Component):
        self._component = component

    def operation(self):
        return self._component.operation()

class ConcreteDecoratorA(Decorator):
    def operation(self):
        return f"ConcreteDecoratorA({self._component.operation()})"

class ConcreteDecoratorB(Decorator):
    def operation(self):
        return f"ConcreteDecoratorB({self._component.operation()})"

# 测试装饰器模式
simple = ConcreteComponent()
print(f"Client: I've got a simple component:\n{simple.operation()}")

decorator1 = ConcreteDecoratorA(simple)
decorator2 = ConcreteDecoratorB(decorator1)
print(f"Client: Now I've got a decorated component:\n{decorator2.operation()}")

四、行为型模式

策略模式(Strategy)
策略模式定义了一系列算法,并将每一个算法封装起来,使它们可以相互替换。策略模式让算法独立于使用它的客户而变化。

代码示例:

from abc import ABC, abstractmethod

class Strategy(ABC):
    @abstractmethod
    def do_algorithm(self, data: list):
        pass

class ConcreteStrategyA(Strategy):
    def do_algorithm(self, data: list):
        return sorted(data)

class ConcreteStrategyB(Strategy):
    def do_algorithm(self, data: list):
        return sorted(data, reverse=True)

class Context:
    def __init__(self, strategy: Strategy):
        self._strategy = strategy

    def set_strategy(self, strategy: Strategy):
        self._strategy = strategy

    def do_some_business_logic(self):
        print("Context: Sorting data using the strategy")
        result = self._strategy.do_algorithm(["a", "b", "c", "d", "e"])
        print(",".join(result))

# 测试策略模式
context = Context(ConcreteStrategyA())
print("Client: Strategy is set to normal sorting.")
context.do_some_business_logic()

print("\nClient: Strategy is set to reverse sorting.")
context.set_strategy(ConcreteStrategyB())
context.do_some_business_logic()



观察者模式(Observer)
观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。当主题对象状态发生变化时,它会通知所有观察者对象,使它们能够自动更新自己。

代码示例:

class Subject:
    _state: int = None
    _observers = []

    def attach(self, observer):
        self._observers.append(observer)

    def detach(self, observer):
        self._observers.remove(observer)

    def notify(self):
        for observer in self._observers:
            observer.update(self)

    def some_business_logic(self):
        self._state = 123
        self.notify()

class Observer:
    def update(self, subject):
        pass

class ConcreteObserverA(Observer):
    def update(self, subject):
        if subject._state == 123:
            print("ConcreteObserverA: Reacted to the event")

class ConcreteObserverB(Observer):
    def update(self, subject):
        if subject._state == 123:
            print("ConcreteObserverB: Reacted to the event")

# 测试观察者模式
subject = Subject()

observer1 = ConcreteObserverA()
subject.attach(observer1)

observer2 = ConcreteObserverB()
subject.attach(observer2)

subject.some_business_logic()

五、结语

今天我们一起探索了几种常见的设计模式,并且提供了相应的 Python 代码示例。设计模式不仅仅是代码的优化手段,更是一种编程思想的升华。希望你能从中学到一些新的知识,并在实际项目中灵活应用这些设计模式,写出更加优雅、高效的代码。

记住,编程不仅仅是一种技能,更是一门艺术。愿你在编程的道路上不断前行,成为一个真正的“代码艺术家”!

在这里插入图片描述

六、作者Info

Author:小鸿的摸鱼日常

Goal:让编程更有趣! 专注于 Web 开发、爬虫,游戏开发,数据分析、自然语言处理,AI 等,期待你的关注,让我们一起成长、一起 Coding!

版权说明:本文禁止抄袭、转载,侵权必究!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1986069.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Redis面试题大全

文章目录 Redis有哪几种基本类型Redis为什么快?为什么Redis6.0后改用多线程?什么是热key吗?热key问题怎么解决?什么是热Key?解决热Key问题的方法 什么是缓存击穿、缓存穿透、缓存雪崩?缓存击穿缓存穿透缓存雪崩 Redis…

python爬虫预备知识三-多进程

python实现多进程的方法:fork、multiprocessing模块创建多进程。 os.fork方法 os.fork方法只适合于unix/linux系统,不支持windows系统。 fork方法调用一次会返回两次,原因在于操作系统将当前进程(父进程)复制出一份…

ESP8266使用舵机以及16路PWM舵机PCA 9685的使用方式

PWM全称 50Hz也就是一秒内变换50次 根据上面的公式 一个高电平一个低电平叫一个脉冲。 例如每个脉冲占20毫秒,那么他的频率是多少? 就是用1去除以他的周期,也就是我们上面说的20,那么就是除0.02,1秒等于1000毫秒,20…

网络安全 - 应急响应检查表

前言 本项目旨在为应急响应提供全方位辅助,以便快速解决问题。结合自身经验和网络资料,形成检查清单,期待大家提供更多技巧,共同完善本项目。愿大家在应急之路一帆风顺。 图片皆来源于网络,如有侵权请联系删除。 一…

南山智尚10亿元定增质疑声连连,与控股股东超70亿资金往来引瞩目

《港湾商业观察》施子夫 王璐 近期,南山智尚(300918.SZ)发布了《向特定对象发行A股股票募集说明书(修订稿)》。 据了解,公司此次拟募集资金总额不超过10亿元,扣除发行费用后的募集资金净额将全部用于年产8万吨高性能…

鸿蒙HarmonyOS实战:ArkUI组件添加内容背景模糊效果

动画效果可以丰富界面的细节,提升UI界面的真实感和品质感。例如,模糊和阴影效果可以让物体看起来更加立体,使得动画更加生动。ArkUI提供了丰富的效果接口,开发者可快速打造出精致、个性化的效果。本章中主要对常用的模糊、阴影和色…

速卖通、Lazada、虾皮卖家是如何自建买家账号测评的?

在跨境电商领域,速卖通、Lazada、Shopee等平台上的卖家为了提升店铺信誉、提高产品排名和销量,常常需要借助买家账号进行测评。然而,依赖外部服务商往往存在风险,如账号质量参差不齐、恶意差评等问题。因此,自己养国外…

Crowd-SAM:SAM作为拥挤场景中目标检测的智能标注器

摘要 目标检测是一项重要任务,广泛应用于各种场景。通常,它需要大量的标签进行训练,这相当耗时,尤其是在拥挤的场景中。最近,Segment Anything Model(SAM)作为一种强大的零样本分割器应运而生&…

数学中常用字母符号读法

文章目录 一、希腊字母二、其他字母1.字母上一横2.拉长的s(‌∫)‌3.数列中的e4. N:非负整数集合或自然数集合{0,1,2,3,…n} 一、希腊字母 Ω ω:欧米伽 Omega。 ∑ σ:西格玛 Sigma。 作用:是一个求和符号&#xf…

滑动窗口专题——找到所有字母的异位词

一、题目解析: 题意如图 二、算法分析 本题依旧是两种方法:暴力枚举、滑动窗口 暴力枚举: 枚举出所有的字串进行比较,符合则记录位置,最终返回结果数组。 滑动窗口哈希表: 思路:1、初始化左…

怎么区分Alpha因子和风险因子?

这是一个絮絮叨叨的专题系列,跟大伙儿唠一唠量化相关的小问题,有感而发写到哪算哪,这是第3期,来唠个14块钱的~ 不知大伙儿有木有这样的疑惑? 看到Barra里面有Size、Liquid等因子,这些因子同样出现在很多人的…

【Redis】String字符串

目录 String字符串 常见命令 SET GET MSET MGET SETNX 计数命令 INCR INCRBY DECY DECYBY INCRBYFLOAT 其他命令 APPEND GETRANGE SETRANGE STRLEN 内部编码 String类型的典型使用场景 缓存(Cache)功能 计数功能 共享会话(Session) String字符串 字符…

bat批处理文件 —— 用于自动化环境配置和项目执行

文章目录 一、什么是 bat ?1.1、支持 bat 的编辑软件1.2、常用命令 三、项目实战3.1、入门案例3.2、(自动化)环境配置与python库安装3.3、将 bat 当成一个简易的 .exe 可执行文件 四、标识符详解4.1、rem:添加注释4.2、echo off&a…

网工内推 | 上市公司IT工程师,最高15薪,周末双休

01 上海索辰信息科技股份有限公司 🔷招聘岗位:IT工程师 🔷岗位职责: 1、熟悉代码研发类企业的内部信息化管理,参与公司自主开发系统的规划和建设,搭建高级别内部信息安全体系,对内部信息的安全…

【practise】删除有序数组中的重复项

关于博主: 今天分享一道简单的关于“双指针”算法的题目。算是双指针中非常基础的题目,有兴趣可以借鉴一波~ 目录 1.题目介绍2.题解思路:双指针法3.代码示例 1.题目介绍 题目链接:LINK 本题要求是:对给定的有序数组…

Android中的沉浸式丝滑转场之共享元素转场动画

文章目录 1. 介绍2. 实现方法3. 举例演示3.1 举例一:普通页面间共享元素转场动画3.2 举例二:列表页面共享元素转场动画 4. 总结 1. 介绍 在Android开发中,经常会有页面转场的动画效果。普通的转场动画不过是左进右出,渐显渐隐&am…

树莓派4B学习笔记24:Python_SYN6288语音模块的控制函数

今日继续学习树莓派4B 4G:(Raspberry Pi,简称RPi或RasPi) 本人所用树莓派4B 装载的系统与版本如下: 版本可用命令 (lsb_release -a) 查询: ​ Opencv 版本是4.5.1: ​ Python 版本3.7.3: 今日学习SYN6288语…

叛逆,批判

1、对以往说法的批判之一(第一次这么公开批判是2004-2005年): 这部英文版的《数学百科全书》似乎是从俄语版翻译过来的?我查了三本引用的图书文献,都没有关于“nonsingular”和“singular”的类似下面的说法&#xff…

加密市场再遭重创:多重利空因素引发超10亿美元抛售潮

四年前的全球“熔断潮”仿佛还在眼前,如今金融市场再度迎来剧烈震荡,全球股市与加密货币市场遭遇多重利空冲击,尤其是比特币和以太坊的闪崩使得市场恐慌情绪蔓延。这个“黑色星期一”不仅见证了股市的跳水,还标志着加密市场的又一…

每日学习笔记:C++ STL算法之已排序区间算法

目录 查找元素 检查某个元素是否存在: binary_search(beg, end, value) binary_search(beg, end, value, op) 检查数个元素是否存在: includes(beg, end, searchBeg, searchEnd) includes(beg, end, searchBeg, searchEnd, op) ​编辑 查找第一个或最后一…