Python面向对象 | 抽象类和接口类
一、抽象类(规范的编程模式)
什么是抽象类
抽象类是一个特殊的类,它的特殊之处在于只能被继承,不能被实例化。抽象类的本质还是类,指的是一组类的相似性,而接口只强调函数属性的相似性。
为什么要有抽象类
如果说类是从一堆对象中抽取相同的内容而来的,那么抽象类就是从一堆类中抽取相同的内容而来的,内容包括数据属性和函数属性。
抽象类是介于类和接口之间的一个概念,同时具备类和接口的部分特性,可以用来实现归一化设计。在python中,并没有接口类这种东西,即便不通过专门的模块定义接口,我们也应该有一些基本的概念
从设计角度去看,如果类是从现实对象抽象而来的,那么抽象类就是基于类抽象而来的。
从实现角度来看,抽象类与普通类的不同之处在于:抽象类中有抽象方法,该类不能被实例化,只能被继承,且子类必须实现抽象方法。这一点与接口有点类似,但其实是不同的,即将揭晓答案
抽象类就是制定一个规则,让其他人按照我的规则写程序。 约定俗称规范,但是有人不按照这个执行。
归一化设计 强制指定规范.
写一个抽象类:
- from abc import ABCMeta,abstractmethod
- 需要用到abc模块,在这个类创建的时候指定 metaclass = ABCMeta
- 在你希望子类实现的方法上加上 @abstramethod 装饰器
举例,支付宝和qq支付,统一支付方式:实例化的时候,有不规范的,直接报错呢
from abc import ABCMeta,abstractmethod
class Payment(metaclass=ABCMeta): # 抽象类 接口类 规范和约束 metaclass指定的是一个元类
"""
此类什么都不做,就是制定一个标准,谁继承我,必须定义我里面的方法。
"""
@abstractmethod
def pay(self):
pass
class Alipay(Payment):
def pay(self,money):
print('使用支付宝支付了%s元' % money)
class Wechatpay(Payment):
def pay(self,money):
print('使用微信支付了%s元' % money)
class ApplePay(Payment):
def pay(self,money):
print('使用applepay支付了%s元' % money)
def pay(object,money): # 归一化设计
object.pay(money) # 执行实例化对象的类方法
# p = Payment() #抽象类不能实例化,TypeError: Can't instantiate abstract class Payment with abstract methods pay
a = Alipay()
a.pay(100)
pay(a,100) # 上下效果相同
# Payment不需要pay的代码,只需要子类实现即可
# Payment没有任何代码实现,必须要求子类有同名的方法名
'''
执行输出:
使用支付宝支付了100元
使用支付宝支付了100元
'''
这个统一的方法,叫做归一化设计
归一化设计:不管是哪一个类的对象,都调用同一个函数去完成相似的功能
归一化使得高层的外部使用者可以不加区分的处理所有接口兼容的对象集合
抽象类和接口类做的事情 :建立规范
制定一个类的metaclass是ABCMeta,那么这个类就变成了一个抽象类(接口类),这个类的主要功能就是建立一个规范
抽象类中所有被abstractmethod装饰的方法都必须被继承的子类实现。如果不实现,那么在实例化阶段将会报错
无论是抽象类还是接口类metaclass=ABCMeta 都不可以被实例化。p = Payment() 报错
如果抽象类没有abstractmethod装饰器,那么这个方法,子类不需要实现。把fuqian改成pay就可以实例化了
二、接口类
接口类:定义一个接口对继承类进行约束,接口里有什么方法,继承类就必须有什么方法,接口中不能有任何功能代码
比如动物园里面的动物,会游泳,走路,爬树,飞行的动物。定义几个动物:
class Tiger(object): # 老虎
def walk(self):pass # 走路
def swim(self):pass # 游泳
class Monkey(object): # 猴子
def walk(self):pass
def climb(self):pass # 爬树
class Swan(object): # 天鹅
def walk(self): pass
def swim(self): pass
def fly(self):pass # 飞行
def cal_flying_speed(self):pass # 计算飞行速度
def cal_flying_height(self):pass # 计算飞行高度
观察上面的代码,技能重复了。这样写,容易丢失方法。加一个鹦鹉,但是它少了一些飞行类的方法
class Parrot(object):
def fly(self):pass
def cal_flying_speed(self): pass
怎么解决这个问题呢?定义一个抽象类A
from abc import ABCMeta, abstractmethod
class A(metaclass=ABCMeta):
@abstractmethod
def fly(self): pass
@abstractmethod
def cal_flying_speed(self): pass
@abstractmethod
def cal_flying_height(self): pass
class Tiger(object): # 老虎
def walk(self): pass # 走路
def swim(self): pass # 游泳
class Monkey(object): # 猴子
def walk(self): pass
class Swan(A): # 天鹅
def walk(self): pass
def swim(self): pass
def fly(self): pass # 飞行
def cal_flying_speed(self): pass # 计算飞行速度
def cal_flying_height(self): pass # 计算飞行高度
class Parrot(A):
def fly(self): pass
def cal_flying_speed(self): pass
Swan() # 没问题
Parrot() # 报错
'''
执行报错:
TypeError: Can't instantiate abstract class Parrot with abstract methods cal_flying_height
'''
鹦鹉实例化时,报错,找不到方法cal_flying_height。这样就约束了飞行动物的方法。所有会走的动物,具有一些会走的动物特性。对于爬行动物,不能继承A,所以需要再定义一抽象类
from abc import ABCMeta,abstractmethod
class FlyAnimal(metaclass=ABCMeta): # 飞行
@abstractmethod
def fly(self):pass # 接口中的方法函数不能有任何内容
@abstractmethod
def cal_flying_speed(self):pass
@abstractmethod
def cal_flying_height(self):pass
class WalkAnimal(metaclass=ABCMeta): # 走路
@abstractmethod
def walk(self):pass
class SwimAnimal(metaclass=ABCMeta): # 游泳
@abstractmethod
def swim(self):pass
class Tiger(WalkAnimal,SwimAnimal): # 老虎,继承走路和游泳
def walk(self):pass # 走路
def swim(self):pass # 游泳
class Monkey(WalkAnimal): # 猴子
def walk(self):pass
def climb(self):pass # 爬树
class Swan(FlyAnimal,WalkAnimal,SwimAnimal): # 天鹅,继承飞行,走路,游泳
def walk(self): pass
def swim(self): pass
def fly(self):pass # 飞行 #调用时,要完成代码
def cal_flying_speed(self):pass # 计算飞行速度
def cal_flying_height(self):pass # 计算飞行高度
class Parrot(FlyAnimal): # 鹦鹉,继承飞行
def fly(self):pass
def cal_flying_speed(self): pass
def cal_flying_height(self): pass
#实例化
Tiger()
Monkey()
Swan()
Parrot()
执行输出,就没有报错了。
接口隔离原则:
使用多个专门的接口,而不使用单一的总接口。即客户端不应该依赖那些不需要的接口。
不能使用单一的总接口来完成,所以需要定义多个抽象类,同时,不需要的接口不要给底层类继承。
抽象类和接口类总结:
在python中,并没有什么不同,都是用来约束子类中的方法的;
只要是抽象类和接口类中被abstractmethod装饰的方法,都需要被子类实现;
当多个类有相同的功能也有不同的功能的时候,应该采用多个接口类来进行分别的约束
面试的时候,可能会问:什么是抽象类?什么是接口类?
抽象类 是python中定义类的一种规范,用来约束子类中的方法的。被abstractmethod装饰的方法,子类必须实现,否则实例化时报错。
接口类 满足接口隔离原则,且完成多继承的约束。如果不按照规范,在调用方法时,报错。
Python面向对象 | 抽象类和接口类的更多相关文章
- python基础 抽象类(接口类)
Python中没有接口.接口类,抽象类:定义 制定一个规范 #必须要导入from abc import ABCMeta,abstractmethod class Payment(metaclass = ...
- python开发面向对象基础:接口类&抽象类&多态&钻石继承
一,接口类 继承有两种用途: 一:继承基类的方法,并且做出自己的改变或者扩展(代码重用) 二:声明某个子类兼容于某基类,定义一个接口类Interface,接口类中定义了一些接口名(就是函数名)且并未实 ...
- python之路----继承的抽象类和接口类
抽象类与接口类 接口类 继承有两种用途: 一:继承基类的方法,并且做出自己的改变或者扩展(代码重用) 二:声明某个子类兼容于某基类,定义一个接口类Interface,接口类中定义了一些接口名(就是函数 ...
- Python抽象类和接口类
一.抽象类和接口类 继承有两种用途: 一:继承基类的方法,并且做出自己的改变或者扩展(代码重用) 二:声明某个子类兼容于某基类,定义一个接口类Interface,接口类中定义了一些接口名(就是函数名) ...
- python's twenty-first day for me 抽象类和接口类以及多态
归一化设计: 不管是哪一个类的对象,都调用同一个函数去完成相似的功能. class Alipay: def pay(self,money): print('使用支付宝支付了%s' % money) c ...
- 抽象类,接口类,封装,property,classmetod,statimethod
抽象类,接口类,封装,property,classmetod,statimethod(类方法,静态方法) 一丶抽象类和接口类 接口类(不崇尚用) 接口类:是规范子类的一个模板,只要接口类中定义的,就应 ...
- php 抽象类和接口类
PHP中抽象类和接口类都是特殊类,通常配合面向对象的多态性一起使用. 相同: ①两者都是抽象类,都不能实例化. ②只有接口类的实现类和抽象类的子类实现了 已经声明的 抽象方法才能被实例化. 不同: ① ...
- python 使用abc实现接口类/虚类(2.2)
python 使用abc实现接口类/虚类 具体类 class BaseA: def run(self): print('base A running') class ChildA(BaseA): de ...
- python面向对象 : 抽象类(接口类),多态,封装(私有制封装)
一. 抽象类(接口类) 与java一样, python也有抽象类的概念但是同样需要借助模块实现,抽象类是一个特殊的类, 它的特殊之处在于只能被继承, 不能被实例化. 从设计角度去看, 如果类是从现实对 ...
随机推荐
- c# CRC16位校验辅助类
public class CRC16Helper { /// <summary> /// CRC校验 /// </summary> /// <param name=&qu ...
- mac下mysql安装、卸载、基本操作
2018-06-08 10:57:26 张广森 阅读数 6584更多 分类专栏: mac mysql 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本 ...
- 【转】PyQt弹出式对话框的常用方法及标准按钮类型
pyQt之弹出式对话框(QMessageBox)的常用方法及标准按钮类型 一.控件说明 QMessageBox是一种通用的弹出式对话框,用于显示消息,允许用户通过单击不同的标准按钮对消息进行反馈,且每 ...
- c#的IDisposable
尽量在using中使用非托管资源 1.实现Dispose方法 2.提取一个受保护的Dispose虚方法,在该方法中实现具体的释放资源的逻辑 3.添加析构函数 4.添加一个私有的bool类型的字段,作为 ...
- 【2】hexo+github搭建个人博客的简单使用
使用hexo+github搭建一个可以外网访问的个人博客,此文用于记录博客初级的使用方法. 新建-编写-生成-部署文章的全过程 1.使用cmd完成 打开命令提示符[win+r输入cmd] 切换到自己本 ...
- Centos 7.6 安装 oracle 10.2.0.1 数据库软件
step 1: 编辑 /etc/redhat-release :内容为redhat-4 step 2: 安装32位的软件包:yum install libXp.i686 libXt.i686 li ...
- Blend 设置一个圆形的按钮
原文:Blend 设置一个圆形的按钮 1)画一个圆形 右击构成控件 3)选择button 当然如果想做成别的控件 都可以 4)我们有了一个button 5)做动画 6)定义触发器 7)定义事件 效果
- table中td文字超出长度用省略号隐藏超出内容,鼠标点击内容全部显示
1,设置css样式 <style>table {width: 100%;float: left;table-layout:fixed;width:600px;border:1px soli ...
- vs2017 + mysql+ef 创建实体模型闪退问题
=>:需要下载安装三个包 mysql和VS链接的驱动 vs2017推荐安装版本:mysql-connector-net-6.10.7.msi vs2015推荐安装版本: mysql-connec ...
- 1 js中常用的操作
总结一些前端js常用的操作 常用日期操作:前端开发常用 JS 方法 js中array.list.map的遍历:js遍历集合(Array,Map,Set) js中对象的定义:js创建自定义对象的几种方式 ...