Python抽象基类之声明协议
抽象基类之--声明协议
上回讲了Python中抽象基类的大概,相信大家对abcmeta
以及什么是抽象基类已经有所了解。传送门
现在我们来讲讲抽象基类的另一个常用用法--声明协议
所谓声明协议,有点像Java中的接口这个概念。就是子类必须实现父类要求的方法。
1.不使用抽象基类来实现
1. 提出异常(规范起见请使用 NotImplementedError)
class Test(object):
def __init__(self):
self.salary = []
def work(self):
money = self._job()
self.salary.append(money)
return money
def _job(self):
raise NotImplementedError("Test子类必须实现 '_job' 方法")
如上所示的类,如果子类没有自己重写_job
方法而执行work
方法就会产生NotImplementedError
异常,这样通过异常的方法就可以强迫子类必须实现父类规定的函数,但是这有个缺点:如果子类不调用work
方法,那么子类就算没有实现_job
方法也不会报错。
2. 使用元类
通过元类也可以实现声明协议
的功能
class TestMeta(type):
def __new__(cls, name, bases, attrs):
# 首先判断该类是不是基类(通过手动定义 abstract 属性来判定)
if attrs.pop("abstract", False):
return super(TestMeta, cls).__new__(cls, name, bases, attrs)
# 否则就是子类,就要判断子类有没有一个叫 _job 的方法
_job = attrs.get("_job", None)
if _job and callable(_job):
return super(TestMeta, cls).__new__(cls, name, bases, attrs)
# 子类为定义 _job 方法
raise TypeError("Test子类必须要定义 _job 方法")
class Test(metaclass=TestMeta):
abstract = True
def __init__(self):
self.salary = []
def work(self):
money = self._job()
self.salary.append(money)
return money
# 这个类会报错
# class SubTestA(Test):
# pass
class SubTestB(Test):
def _job(self):
return 3000
上面的例子中,我们通过元类来检查创建的子类是否包含有_job
方法。相比起方法一,该方法必须要求子类定义_job
类无论有没有调用过_job
方法。但是与Java中的接口相比又少了一点什么?Java中的接口是不可以直接实例化对象的,但是在这个例子中我们可以直接创建Test的对象。这时就该抽象基类上场了。
2.抽象基类实现声明协议
我们可以使用abc
模块中的一个装饰器--abstractmethod
来装饰一个类的方法使其成为一个抽象方法,拥有抽象方法的类就不可以被实例化(像不像C++中的虚函数)举个例子。
import abc
class Test(metaclass=abc.ABCMeta):
def __init__(self):
self.salary = []
def work(self):
money = self._job()
self.salary.append(money)
return money
@abc.abstractmethod
def _job(self):
raise NotImplementedError("Test子类必须实现 '_job' 方法")
# t = Test() # 报错: TypeError: Can't instantiate abstract class Test with abstract methods _job
# 子类可以被创建,但是不能初始化
class SubTestA(Test):
pass
# sta = SubTestA() # 报错: TypeError: Can't instantiate abstract class SubTestA with abstract methods _job
# 实现抽象函数的子类可以被实例化
class SubTestB(Test):
def _job(self):
return 3000
stb = SubTestB()
对比一下上述三种方法的区别:
异常 | 元类 | 抽象基类 |
---|---|---|
在调用方法时产生异常 | 在定义类时产生异常 | 在实例化对象时产生异常 |
Python抽象基类之声明协议的更多相关文章
- Python抽象基类:ABC谢谢你,因为有你,温暖了四季!
Python抽象基类:ABC谢谢你,因为有你,温暖了四季! Python抽象基类:ABC谢谢你,因为有你,温暖了四季! 实例方法.类方法和静态方法 抽象类 具名元组 参考资料 最近阅读了<Pyt ...
- python抽象基类
抽象基类 抽象基类提了一种方式,用以组织对象的层次结构,做出关于所需方法的断言,以及实现其他一些功能 要定义抽象基类,需要使用abc模块,该模块定义了一个元类(ABCMeta) 和一组装饰器(@abs ...
- NSObject class和NSObject protocol的关系(抽象基类与协议)
[转载请注明出处] 1.接口的实现 对于接口这一概念的支持,不同语言的实现形式不同.Java中,由于不支持多重继承,因此提供了一个Interface关键词.而在C++中,通常是通过定义抽象基类的方式来 ...
- 4.6 C++抽象基类和纯虚成员函数
参考:http://www.weixueyuan.net/view/6376.html 总结: 在C++中,可以通过抽象基类来实现公共接口 纯虚成员函数没有函数体,只有函数声明,在纯虚函数声明结尾加上 ...
- Python 接口:从协议到抽象基类
p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px Helvetica } 抽象基类的常见用途:实现接口时作为超类使用.然后,说明抽象基类如何检查 ...
- 【Python】【元编程】【从协议到抽象基类】
"""class Vector2d: typecode = 'd' def __init__(self,x,y): self.__x = float(x) self.__ ...
- guxh的python笔记七:抽象基类
1,鸭子类型和白鹅类型 1.1,白鹅类型 白鹅类型对接口有明确定义,比如不可变序列(Sequence),需要实现__contains__,__iter__,__len__,__getitem__,__ ...
- Fluent_Python_Part4面向对象,11-iface-abc,协议(接口),抽象基类
第四部分第11章,接口:从协议到抽象基类(重点讲抽象基类) 接口就是实现特定角色的方法集合. 严格来说,协议是非正式的接口(只由文档约束),正式接口会施加限制(抽象基类对接口一致性的强制). 在Pyt ...
- 流畅python学习笔记:第十一章:抽象基类
__getitem__实现可迭代对象.要将一个对象变成一个可迭代的对象,通常都要实现__iter__.但是如果没有__iter__的话,实现了__getitem__也可以实现迭代.我们还是用第一章扑克 ...
随机推荐
- Pytorch 分割模型构建和训练【直播】2019 年县域农业大脑AI挑战赛---(四)模型构建和网络训练
对于分割网络,如果当成一个黑箱就是:输入一个3x1024x1024 输出4x1024x1024. 我没有使用二分类,直接使用了四分类. 分类网络使用了SegNet,没有加载预训练模型,参数也是默认初始 ...
- redis api-zset
- CSP-S2019 爆炸记
DAY -1 停课的第五天.早上来机房教练居然不在,先看了一道憨题,发现ST表+二分查找nlogn水过,然后发现单调栈可以O(n),肥肠开心 打了走人. 然后就开始颓了(逃 颓了一会之后看愤怒的小鸟这 ...
- go语言的基础类型
1.布尔类型:bool 2.整型:int8,byte,int16,int,uint,uintptr等 3.浮点类型:float32.float64 4.复数类型:complex64,complex12 ...
- DevOps - 自动化工具
章节 DevOps – 为什么 DevOps – 与传统方式区别 DevOps – 优势 DevOps – 不适用 DevOps – 生命周期 DevOps – 与敏捷方法区别 DevOps – 实施 ...
- 2-10 就业课(2.0)-oozie:11、hadoop的federation(联邦机制,了解一下)
==================================================== Hadoop Federation 背景概述 单NameNode的架构使得HDFS在集群扩展性 ...
- CLion的使用
配置远程Linux编译器 实现目标:1.将项目中的源码和target和Linux服务器同步.2.代码在服务器端运行 配置ToolChains setting -> Build,Execution ...
- 把自己的项目发布到maven仓库并在maven和gradle中开始使用
把自己的项目发布到maven仓库并在maven和gradle中开始使用 上一条博客中提到的日志打印项目总算是维护的差不多了, 不过现在使用它还是打成jar包放到其他项目内, 所以决定把项目传到mave ...
- PL/SQL 找到某列都为空的列名
DECLARE CURSOR temp IS SELECT COLUMN_NAME FROM ALL_TAB_COLUMNS WHERE TABLE_NAME=Upper('xxx'); v_num ...
- POJ 3468 区间更新(求任意区间和)A Simple Problem with Integers
A Simple Problem with Integers Time Limit: 5000MS Memory Limit: 131072K Total Submissions: 163977 ...