抽象基类之--声明协议

上回讲了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抽象基类之声明协议的更多相关文章

  1. Python抽象基类:ABC谢谢你,因为有你,温暖了四季!

    Python抽象基类:ABC谢谢你,因为有你,温暖了四季! Python抽象基类:ABC谢谢你,因为有你,温暖了四季! 实例方法.类方法和静态方法 抽象类 具名元组 参考资料 最近阅读了<Pyt ...

  2. python抽象基类

    抽象基类 抽象基类提了一种方式,用以组织对象的层次结构,做出关于所需方法的断言,以及实现其他一些功能 要定义抽象基类,需要使用abc模块,该模块定义了一个元类(ABCMeta) 和一组装饰器(@abs ...

  3. NSObject class和NSObject protocol的关系(抽象基类与协议)

    [转载请注明出处] 1.接口的实现 对于接口这一概念的支持,不同语言的实现形式不同.Java中,由于不支持多重继承,因此提供了一个Interface关键词.而在C++中,通常是通过定义抽象基类的方式来 ...

  4. 4.6 C++抽象基类和纯虚成员函数

    参考:http://www.weixueyuan.net/view/6376.html 总结: 在C++中,可以通过抽象基类来实现公共接口 纯虚成员函数没有函数体,只有函数声明,在纯虚函数声明结尾加上 ...

  5. Python 接口:从协议到抽象基类

    p.p1 { margin: 0.0px 0.0px 0.0px 0.0px; font: 15.0px Helvetica } 抽象基类的常见用途:实现接口时作为超类使用.然后,说明抽象基类如何检查 ...

  6. 【Python】【元编程】【从协议到抽象基类】

    """class Vector2d: typecode = 'd' def __init__(self,x,y): self.__x = float(x) self.__ ...

  7. guxh的python笔记七:抽象基类

    1,鸭子类型和白鹅类型 1.1,白鹅类型 白鹅类型对接口有明确定义,比如不可变序列(Sequence),需要实现__contains__,__iter__,__len__,__getitem__,__ ...

  8. Fluent_Python_Part4面向对象,11-iface-abc,协议(接口),抽象基类

    第四部分第11章,接口:从协议到抽象基类(重点讲抽象基类) 接口就是实现特定角色的方法集合. 严格来说,协议是非正式的接口(只由文档约束),正式接口会施加限制(抽象基类对接口一致性的强制). 在Pyt ...

  9. 流畅python学习笔记:第十一章:抽象基类

    __getitem__实现可迭代对象.要将一个对象变成一个可迭代的对象,通常都要实现__iter__.但是如果没有__iter__的话,实现了__getitem__也可以实现迭代.我们还是用第一章扑克 ...

随机推荐

  1. JS弹出层制作,以及移动端禁止弹出层下内容滚动,overflow:hidden移动端失效问题

    HTML <div class="layer"> <div class="menu-list"> <span>社会</ ...

  2. SpringBoot Date类型插入数据库始终比正确时间早一天问题解决办法

    bug描述 昨天的Date插入不进去问题解决后,一直没发现其实插入的时间一直比正确的时间早一天 输出sql语句,发现insert语句还是对的,不知道为什么插入数据库之后结果就早了一天 https:// ...

  3. 引用类型--Object类型、Array类型

    引用类型的值(对象)是引用类型的一个实例.在ECMAScript中,引用类型是一种数据结构,它描述的是一类对象具有的属性和方法. 对象是某个特定引用类型的实例,新对象是使用new操作符后跟一个构造函数 ...

  4. 「USACO08JAN」电话线Telephone Lines

    传送门 Luogu 解题思路 考虑二分,每次把大于二分值的边的权设为1,小于等于的设为0,如果最短路<=k则可行,记得判无解 细节注意事项 咕咕咕 参考代码 #include <algor ...

  5. Gradient descend 梯度下降法和归一化、python中的实现(未完善)

    梯度下降法是优化函数参数最常用.简单的算法 通常就是将一组输入样本的特征$x^i$传入目标函数中,如$f(x) = wx + b$,再计算每个样本通过函数预测的值$f(x^i)$与其真实值(标签)$y ...

  6. 关于Burp Suite Intruder 的四种攻击方式

    以下面这一段参数为例,被§§包围的部分为需要破解的部分: user=§ss§&password=§zxcv§&imageField.x=17&imageField.y=1 (1 ...

  7. python如何输出矩阵的行数与列数?

    Python如何输出矩阵的行数与列数? 对于pyhton里面所导入或者定义的矩阵或者表格数据,想要获得矩阵的行数和列数有以下方法: 1.利用shape函数输出矩阵的行和列 x.shape函数可以输出一 ...

  8. logback日志

    一.什么是日志框架? 是一套能够实现日志输出的工具包 能够描述系统运行状态的所有时间都可以算作日志 用户下线,接口超时,数据崩溃 二.日志框架的能力 1.定制输出目标(文件,回滚策略,数据库,网络的第 ...

  9. 图片转换到指定大小PDF

    1.首先转换为eps jpeg2ps compile to exec file ./jpeg2ps  -p a4  a.jpg -o x.eps2.从eps转换到pdf ps2pdf -dDownsa ...

  10. Lua 完美打印数据 (例子)

    例子1 : ableprint = function(data,cstring,deepIndex) --第二个参数可以为空,第三个参数不要手动添加,它是用来进行打印深度控制的. if data == ...