接口(python 中的协议)的多种不同的实现方式即为多态。多态的作用,就是为了类在继承和派生的时候,保证使用“家谱”中任一类的实例的某一属性时的正确调用。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
|
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++中的纯虚函数类
这个类只能当做协议(接口)类
总结
本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注服务器之家的更多内容!
原文链接:https://blog.csdn.net/qq_42031243/article/details/121549953