Python 单例模式工厂模式和classmethod装饰器

ops/2024/12/16 5:10:13/

前言:

Python作为面向对象的语言,显然支持基本的设计模式。也具备面向对象的语言的基本封装方法:属性、方法、继承、多态等。但是,做为强大的和逐渐发展的语言,python也有很多高级的变种方法,以适应更多的场景。

我们都知道单例和工厂模式在设计模式中,其实就是一种怪胎。其实就是他们都不是经典的面向对象能够覆盖的场景。

例如,单例,他其实就是非常适配于一个硬件的操控的用例。而工程模型,主要关注的不同的对象的多态的变化。这两张,从根本上讲是从典型的面向对象的体系里面分离出来的典型的普通的例子。

由此,Python作为一个语言也发展出相对应的的适配的额外的方法,就如,我们这章将展示的方法。


1 设计模式的基本概念:

1.1 单例模式

单例模式确保一个类只有一个实例,并提供一个全局访问点。

class Singleton:_instance = Nonedef __new__(cls):if cls._instance is None:cls._instance = super(Singleton, cls).__new__(cls)return cls._instance# 使用单例模式
singleton1 = Singleton()
singleton2 = Singleton()print(singleton1 is singleton2)  # 输出 True,表示两个变量指向同一个实例

【案,这里创建了两个实例, singleton1 和singleton2 ,这两个不同名字的类实例,其实是一个东西。如果我们用类来代表一个硬件,比如机器视觉的相机或者是PLC,这些都是硬件实体。我们需要用这个实例模式来定义,以避免同一个硬件资源被其他的的任务复用】

1.2 工厂模式:

工厂模式是一种创建型设计模式,它使用工厂方法来创建对象,而不是直接调用构造函数。

class Car:def __init__(self, model, color):self.model = modelself.color = colordef speak(self):return f"This is a {self.model} in {self.color}."class ElectricCar(Car):def speak(self):return f"This is an electric {self.model} in {self.color}."class SportsCar(Car):def speak(self):return f"This is a sports {self.model} in {self.color}."class CarFactory:def create_car(self, model, color, car_type):if car_type == "electric":return ElectricCar(model, color)elif car_type == "sports":return SportsCar(model, color)else:return Car(model, color)# 使用工厂模式创建不同类型的汽车
factory = CarFactory()
my_car = factory.create_car("Toyota Camry", "red", "normal")
my_electric_car = factory.create_car("Nissan Leaf", "blue", "electric")
my_sports_car = factory.create_car("Porsche 911", "yellow", "sports")print(my_car.speak())          # 输出 "This is a Toyota Camry in red."
print(my_electric_car.speak()) # 输出 "This is an electric Nissan Leaf in blue."
print(my_sports_car.speak())    # 输出 "This is a sports Porsche 911 in yellow."

【案,在这个例子中,CarFactory 类的 create_car 方法根据传入的 car_type 参数来决定创建哪种类型的 Car 实例。这样,如果未来需要添加新的汽车类型,比如 HybridCar,你只需要在 CarFactory 类中添加相应的逻辑,而不需要修改其他使用汽车对象的代码。这就是工厂模式的价值所在:它封装了对象的创建逻辑,使得代码更加模块化和易于扩展。】


2 classmethod的装饰器和staticmethod装饰器:

前面的例子里面,我们看到了传统的单例和工厂模式的引入。传统的单例和工厂模式的设计,比较方便的是继承和多态。通过实例,比较方便的是构建继承和多态的方法。但,主要的问题就是需要考虑构建实例后才能使用,这样对多个实例支持的硬件案例,就很麻烦。

2.1 classmethod

在Python中,@classmethod是一个装饰器,用于将一个方法定义为类方法(class method)。类方法与普通的方法(instance method)不同,它不依赖于类的实例(instance),而是依赖于类本身。这意味着类方法的第一个参数总是指向类本身,通常命名为cls

【方法,我们知道面向对象的语言里面,方法就是对象能做哪些事情。一般,方法的定义依赖于类创建的实例,也就是先创建实例,类的方法才有用。而classmethod,是创建了一个类方法。】

类方法的主要作用包括:

  1. 访问类变量类方法可以直接访问类的变量,而不需要创建类的实例。

  2. 创建实例类方法可以用来创建类的实例,这在工厂模式中很常见。

  3. 修改类状态类方法可以修改类的状态,比如添加类变量或者修改类属性。

  4. 继承时保持一致性:在继承时,类方法可以确保子类能够访问父类的类方法

  5. 实现单例模式类方法可以用来实现单例模式,确保一个类只有一个实例。

【案,类的方法有别于传统定义的方法,其实就是提供一个直接方法类定义的途径,以避免之前构建单例模式和工厂那种形而上学的方式(比如前面的单例构建,其实通过类方法就很简洁,不需要构造一个单例】

class MyClass:count = 0  # 类变量@classmethoddef increment_count(cls):cls.count += 1@classmethoddef get_count(cls):return cls.count# 使用类方法
MyClass.increment_count()
print(MyClass.get_count())  # 输出 1

上面的代码,就是在类里面创立了类方法,increment_count,这样直接用类的名字+类方法就可以完成对类的控制了。

这一点,对单例模式(硬件代表)尤其方便,我本来类就是代表硬件特性,采用了类方法,就可以直接方便的访问这些硬件的功能了。

 【案,值得注意的是,get_count的方法的第一个参数cls,是类Myclass本身。】cls = class .

现在,我们把第一节里面的工厂模式,用classmethod的类方法进行修改。

class Animal:def speak(self):raise NotImplementedError("Subclasses must implement this method")class Dog(Animal):def speak(self):return "Woof!"class Cat(Animal):def speak(self):return "Meow!"class AnimalFactory:@classmethoddef get_animal(cls, animal_type):if animal_type == "dog":return Dog()elif animal_type == "cat":return Cat()else:raise ValueError("Unknown animal type")# 使用工厂模式
dog = AnimalFactory.get_animal("dog")
cat = AnimalFactory.get_animal("cat")print(dog.speak())  # 输出 "Woof!"
print(cat.speak())  # 输出 "Meow!"

 在这个示例中,AnimalFactory 类的 get_animal 方法被定义为一个类方法,使用 @classmethod 装饰器。这意味着 get_animal 方法的第一个参数是类本身,这里我们命名为 cls。在这个方法中,我们根据传入的 animal_type 参数来创建相应的动物实例。使用 @classmethod 的好处是,我们可以在不创建类实例的情况下调用这个方法,这在工厂模式中是常见的做法,因为工厂方法通常不需要类的特定实例,而是需要类本身来创建新实例。


2.2 staticmethod

class Dog:def speak(self):return "Woof!"class Cat:def speak(self):return "Meow!"class AnimalFactory:@staticmethoddef get_animal(animal_type):if animal_type == "dog":return Dog()elif animal_type == "cat":return Cat()else:raise ValueError("Unknown animal type")# 使用工厂模式
dog = AnimalFactory.get_animal("dog")
cat = AnimalFactory.get_animal("cat")print(dog.speak())  # 输出 "Woof!"
print(cat.speak())  # 输出 "Meow!"

【案,在工厂模式里面,我们创建了猫和狗两个类,然后,我们没有用猫和狗的类来创建实例。而是通过动物的类,让后,通过动物类来调用猫和狗的类的实例的创建。】

使用 @staticmethod 的缺点:

  1. 灵活性:静态方法不能被子类重写,这限制了它们的灵活性。如果你需要在子类中改变创建实例的行为,静态方法不是一个好选择。

  2. 继承:由于静态方法不属于类的MRO(方法解析顺序),它们不能被继承。

  3. 多态性:静态方法不支持多态,这意味着你不能在子类中重写它们来改变行为。


3 小结:

3.1 self 和 cls

self 参数:

  • self 引用的是类的实例。
  • 用途self 是实例方法的第一个参数,用于访问和修改实例的状态(属性和方法)。
  • 行为self 指向调用方法的实例。当你通过实例调用一个方法时,Python自动将该实例作为 self 参数传递给方法。

cls 参数:

  • cls 引用的是类本身。
  • 用途cls 是类方法的第一个参数,用于访问和修改的状态(类属性和方法),以及创建类的实例
  • 行为cls 指向类本身。类方法可以通过 cls 访问类属性和方法,也可以调用其他类方法,并且可以创建类的实例。

3.1.1 self的例子:

class MyClass:def __init__(self, value):self.value = value  # 使用 self 访问实例属性def print_value(self):print(self.value)  # 使用 self 访问实例属性instance = MyClass(10)
instance.print_value()  # 输出 10

【self,在例子里面作为一个默认的参数,对外是封装的,透明的。在创建类的实例的时候函数init,传递第二个参数的时候,self自动起了衔接类之间的指代和传递的工作,最终,value=50,通过self赋值给了实例里面的类value,self理解为类内部的一个句柄】 

3.1.2 cls的例子:

class MyClass:class_var = "I am a class variable"@classmethoddef print_class_var(cls):print(cls.class_var)  # 使用 cls 访问类属性@classmethoddef create_instance(cls, value):return cls(value)  # 使用 cls 创建实例MyClass.print_class_var()  # 输出 "I am a class variable"
new_instance = MyClass.create_instance(20)
new_instance.print_value()  # 输出 20

【案,上面这个例子很好的说明了cls的使用,结合classmethod,首先,通过create_instance方法+cls,类内部句柄,构造了一个类内部的动态实例构造,cls(value),然后在调用这个创建实例的方法的时候,传入了外部参数value。】

3.2 classmethod(类方法) and staticmethod(静态类)

3.2.1 classmethod(类方法

class MyClass:class_var = "I am a class variable"@classmethoddef my_class_method(cls):print("Class variable:", cls.class_var)MyClass.my_class_method()  # 输出 "Class variable: I am a class variable"

3.2.2 staticmethod

class Car:def __init__(self, model):self.model = modeldef speak(self):return f"This is a {self.model}"class CarFactory:@staticmethoddef create_car(model):return Car(model)# 使用工厂模式
my_car = CarFactory.create_car("Toyota Camry")
print(my_car.speak())  # 输出 "This is a Toyota Camry"

 【案,我们看到这两种方法的主要区别,就是要不要构建实例来访问类方法。当然,类方法也可以构造实例如本文3.1.2,需要写一点构造实例的方法,但是,比静态方法要简洁,此外,也更容易做多态的变换】

对应工厂模式而言:

  • 如果你需要一个简单的工厂方法,不需要访问类属性,也不想被子类重写,那么静态方法是一个很好的选择。
  • 如果你需要在子类中重写工厂方法,或者需要访问类属性来决定如何创建实例,那么类方法是更好的选择。

http://www.ppmy.cn/ops/142297.html

相关文章

Oracle PDB的开启和关闭

[生产环境关闭与开启Oracle PDB] 【运维场景】 在运维Oracle PDB的时候经常要开启和关闭PDB,对关闭和开启PDB的操作要非常熟悉。 【操作方法】 1. PDB的打开与关闭 关闭和开启DB的时候要看DB的警告日志,日志位置(在Oracle用户下查看&…

代码随想录第45天

115.不同的子序列 class Solution:def numDistinct(self, s: str, t: str) -> int:n1 len(s)n2 len(t)dp [[0] * (n1 1) for _ in range(n2 1)]for j in range(n1 1):dp[0][j] 1for i in range(1, n2 1):for j in range(1, n1 1):if t[i - 1] s[j - 1]:dp[i][j]…

ArcGIS字符串补零与去零

我们有时候需要 对属性表中字符串的补零与去零操作 我们下面直接视频教学 下面看视频教学 ArcGIS字符串去零与补零 推荐学习 ArcGIS全系列实战视频教程——9个单一课程组合 ArcGIS10.X入门实战视频教程(GIS思维) ArcGIS之模型构建器(Mod…

数据结构和算法-06线段树-01

线段树 什么是线段树 线段树是一种**[二叉搜索树]**,与[**区间树]**相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树的一个结点 [Segment Tree] is a data structure that stores data about range of elements in nodes as a …

etcd节点扩/缩容

etcd集群节点数量的说明 etcd 是基于 raft算法的分布式键值数据库,生来就为集群化而设计的,由于Raft算法在做决策时需要超半数节点的投票,所以etcd集群一般推荐奇数节点,如3、5或者7个节点构成一个集群。 对于具有 n 个成员的集群…

构建centos docker基础镜像

1、介绍 比较老的版本docker镜像,不太好找,可以尝试自己构建 各版本构建基础镜像方法不太一样,方式也不同,自己尝试,本文只介绍了我自己的尝试 2、构建centos5.11 docker镜像 准备iso文件 (1)安…

Tomcat原理(1)——IDEA实现模拟服务端和客户端的互传

引入 一、什么是Tomcat Tomcat是一个开源的Java Web应用服务器,主要用于运行Java编写的网站和Web应用程序。实质上可以理解为是一个容器,一个用于承载项目的容器。 tomcat有什么作用,最基础来讲,当我们创建一个文件,当…

如何发挥网络爬虫利器phpSpider最大功效

要发挥网络爬虫利器phpSpider的最大功效,可以从以下几个方面入手: 一、基础配置与优化 安装与配置: 确保PHP环境已正确安装,并通过Composer等工具安装phpSpider及其依赖。根据目标网站的特点,合理设置phpSpider的配置…