详解Python为什么不用设计模式

作者:有数可据 时间:2022-03-30 16:17:12 

前言

刚刚看了EuroPython 2017一篇演讲,Why You Don't Need Design Patterns in Python,为什么python不用设计模式。演讲者是STXNEXT的Sebastian Buczynski。

他对设计模式的定义是:

  • 常见问题的通用可复用解决方案

  • 定型的最佳实践

他说设计模式是一种似曾相识(Anology),是一种大纲(Outline),他认为设计模式并不是拿来就能用的。

Singleton

详解Python为什么不用设计模式

第一个是Singleton模式,Singleton的精髓就是任何时候,只有一个类的实例。

《设计模式》里面给出的Singleton代码是

声明:


class Singleton {
public:
static Singleton* Instance();
protected:
Singleton();
private:
static Singleton* _instance;
};

实现:


Singleton* Singleton::_instance = 0;

Sebastian 在 Google 上面找Singleton的Python实现,找到了以下代码:

声明:


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

实现:


one_instance = Singleton()
another_instance = Singleton()
one_instance is another_instance # True

Sebastian指出,照抄C++,当然也可以解决问题,但是在python里面有更好的解决方案。比如,可以用@classmethod。不过,最好的解决方案是直接用module。因为module本身就是唯一的,相当于module就实现了singleton,那么,我们为什么要大费周章,搞一个singleton出来呢?

我回忆了一下,尽管Singleton是最简单的设计模式了,但是,我这么多年一直没用。以前写C#的时候,我用的是静态类,静态类本身就是唯一的,所以我不需要singleton。当然,我看到有人也用C#写了和C++一样的Singleton,但是我觉得解决问题就可以了,没必要为了写设计模式而写设计模式。同样,写VB.net的时候,我直接用的module,也不需要singleton。

结论:当年《设计模式》里面的Singleton模式,是为了只有一个类实例。如果编程语言本身,如python, c#, vb.net,已经提供了这样的能力,就没有必要再用C++的套路了。或者说,设计模式就不需要了。

Facade

详解Python为什么不用设计模式
详解Python为什么不用设计模式

(以上图片来自参考[1])

Facade的基本概念是,子系统用Facade来屏蔽内部的复杂实现。

这时,我们可以把子系统的python文件统一放在一个文件夹里,然后在这个文件夹里放一个__init__.py文件。

详解Python为什么不用设计模式

Command

Command模式把请求封装成对象。

Sebastian认为,在python里面,函数就是一等公民,所以没有必要创建对象。


def command(discount_rate):
some_obj.notify_users_about_discount()

也可以用functools创建command


import functools
command = functools.partial(
some_obj.notify_users_about_discount, discount_rate=0.5
)
command()
# equals to
some_obj.notify_users_about_discount(discount_rate=0.5)

Visitor

Python里面没有接口,没有方法重载。那么怎么实现Visitor呢?

Sebastian指出,可以用@SingleDispatch。


from functools import singledispatch
@singledispatch
def visit(node):
type_name = type(node).__name__
raise AttributeError(f'No handler found for {type_name}')

from ast_nodes import Assign, FunctionDef
@visit.register(Assign)
def visit(node):
pass
@visit.register(FunctionDef)
def visit(node):
pass

我们看到,这里的实现,并没有class。

Decorator

Decorator可以用来扩展一个对象。

它实现的方法是新建一个类,这个类和原来的类属于同一个接口。然后这个类接受一个原来的类的对象,每个方法都调用原来的类的方法。

如果套用c++的《设计模式》,我们有


class OriginalClass:
def get_text(self):
pass
def get_number(self):
pass

class Decorator:
def __init__(self, decorated_obj):
self.decorated_obj = decorated_obj
def get_text(self):
return f'<b>{self.decorated_obj.get_text()}</b>'
def get_number(self):
return self.decorated_obj.get_number()

但是,这里可以用python的__getattr__特性来简化实现。


class Decorator:
def __init__(self, decorated_obj):
self.decorated_obj = decorated_obj
def get_text(self):
return f'{self.decorated_obj.get_text()}'
def __getattr__(self, attr_name):
return getattr(self.decorated_obj, attr_name)

来源:https://blog.csdn.net/juwikuang/article/details/117898815

标签:Python,设计模式
0
投稿

猜你喜欢

  • python使用Apriori算法进行关联性解析

    2022-08-15 13:02:10
  • Python中的jquery PyQuery库使用小结

    2023-05-27 11:08:15
  • 利用python发送和接收邮件

    2022-01-23 10:22:07
  • GO语言不固定参数函数与匿名函数的使用

    2024-02-17 14:42:17
  • 一文教你用Pyecharts做交互图表

    2022-11-20 22:33:38
  • img标签中alt和title属性的正确使用

    2008-01-10 12:59:00
  • sqlserver和oracle中对datetime进行条件查询的一点区别小结

    2024-01-15 16:25:54
  • 浅析python 内置字符串处理函数的使用方法

    2021-07-17 09:23:09
  • 用js格式化金额可设置保留的小数位数

    2024-05-21 10:20:44
  • OpenCV+python实现膨胀和腐蚀的示例

    2021-06-12 10:35:00
  • 如何使用 Flask 做一个评论系统

    2023-12-08 21:04:48
  • python去掉空格的一些常用方式

    2022-02-10 04:45:34
  • Go语言sync.Cond基本使用及原理示例详解

    2023-06-28 07:09:01
  • 使用python连接mysql数据库之pymysql模块的使用

    2024-01-16 13:12:11
  • pyCaret效率倍增开源低代码的python机器学习工具

    2021-01-09 10:30:26
  • Python实现基本数据结构中栈的操作示例

    2021-09-04 10:29:23
  • 利用Python统计Jira数据并可视化

    2023-06-26 00:11:46
  • 分享PHP header函数使用教程

    2023-09-04 12:07:14
  • 使用动画实现微信读书的换一批效果(两种方式)

    2023-10-23 14:30:55
  • Python实现双向链表

    2022-06-12 17:41:34
  • asp之家 网络编程 m.aspxhome.com