python 多态 协议 鸭子类型详解
接口(python 中的协议)的多种不同的实现方式即为多态。多态的作用,就是为了类在继承和派生的时候,保证使用“家谱”中任一类的实例的某一属性时的正确调用。
from abc import ABCMeta, abstractmethod # 鸭子类 class Dock(metaclass=ABCMeta): @abstractmethod def Swimming(self): # 游泳方法协议(接口) pass @abstractmethod # 走路协议(接口) def Walk(self): pass @classmethod def __subclasshook__(cls, C): # 判断是否另一个比较类是否实现了 Swimming Walk 协议, 如果实现了鸭子类的这两个协议, # 那么比较类的类型就是一个鸭子类型 # 当代码执行中如果执行到对象和这个类进行 isinstance 类型判断时会走到这个函数进行判断 for method in ('Swimming', 'Walk'): for B in C.__mro__: if method in B.__dict__: if B.__dict__[method] is None: return NotImplemented break else: return NotImplemented return True # 狗类 class Dog(object): # 实现swimming 协议 def Swimming(self): print("狗会狗刨") # 实现walk 协议 def Walk(self): print("狗会走路") def Eat(self): print("狗喜欢吃骨头") # 乌龟类 class Tortoise(object): # 实现swimming 协议 def Swimming(self): print("乌龟会潜水") # 实现walk 协议 def Walk(self): print("乌龟会走路") def Eat(self): print("乌龟喜欢吃鱼") dog = Dog() tortoise = Tortoise() print(isinstance(dog, Dock)) # True print(isinstance(tortoise, Dock)) # True
可以看到,在上面的代码中,只要实现了 Dock 类中的 swimming 和 Walk 方法,那么这个类就可以被叫做 Dock 类
应用场景 如: for 循环, 在python 中 for 循环只能用于可迭代对象, 那么, 我自己定义的类实现了 __iter__协议(接口),这个实例类就是一个可迭代对象,可以被for 循环使用
python 中定义协议类协议使用 @abstractmethod 装饰器,@abstractmethod 装饰过的类是不能进行初始化的,相对于c++中的纯虚函数类
这个类只能当做协议(接口)类
总结
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!
赞 (0)