抽象类和接口:

java

我们先从java讲起,没有java基础的可以略过。

(挖坑)

python

在python并没有抽象类之说,或者说抽象类=接口类(区别于接口)

继承有两种用途:

一:继承基类的方法,并且做出自己的改变或者扩展(代码重用)

二:声明某个子类兼容于某基类,定义一个接口类Interface,接口类中定义了一些接口名(就是函数名)且并未实现接口的功能,子类继承接口类,并且实现接口中的功能

在python中实现接口类必须借助于 abc模块

abc模块

前言:

面向对象的设计中,抽象类,接口这些必不可少的东西,在python中是如何提现的呢?

python作为一个动态语言,没有强类型的检查,而是以鸭子类型的方式提现,在执行的时候python不严格要求你必须是继承指定的父类而来,只要在调用的时候你有相应的方法和属性就可以了,长的像鸭子你就是鸭子。

也正是基于python这样的特性,python中没有interface的概念。

interface并不是普遍存在的,而是作为一个特例出现在java中,为的是解决多继承的问题。

python支持多继承,自然没有这个需要了。

然而我们难免会有这样的需求:子类必须实现某些指定的方法和属性,否则就抛异常。使用一些笨方法自然是可以达到相同的效果,但是python为我们提供了abc模块。

what is abc:

import abc
print([i for i in dir(abc) if not i.startswith("__")])
['ABC', 'ABCMeta', 'WeakSet', 'abstractclassmethod', 'abstractmethod', 'abstractproperty', 'abstractstaticmethod', 'get_cache_token']

How use abc?

竟然没报错?

from abc import ABCMeta, abstractmethod

class Drawable(metaclass=ABCMeta):

    # @abstractmethod
def size(self):
return 'Drawable size' # @abstractmethod
def draw(self, x, y, scale=1.0):
pass def double_draw(self, x, y):
self.draw(x, y, scale=2.0) d=Drawable()

说明。这个接口类竟然可以实例化。

接下来,我们给这个接口类添加一个abstractmethod方法的时候

class Drawable(metaclass=ABCMeta):

    @abstractmethod
def size(self):
return 'Drawable size' # @abstractmethod
def draw(self, x, y, scale=1.0):
pass def double_draw(self, x, y):
self.draw(x, y, scale=2.0) d=Drawable()

它报错了:TypeError: Can't instantiate abstract class Drawable with abstract methods size

总结:有抽象方法的接口类才是不能实例化。

接下来,我们继续看继承关系

from abc import ABCMeta, abstractmethod

class Drawable(metaclass=ABCMeta):

    # @abstractmethod
def size(self):
return 'Drawable size' # @abstractmethod
def draw(self, x, y, scale=1.0):
pass def double_draw(self, x, y):
self.draw(x, y, scale=2.0) class Cicle(Drawable):
# def size(self):
# return 'Cicle size' # def draw(self, x, y, scale=1.0):
# print(x * scale, y * scale)
def paint(self):
pass c=Cicle()

成功了。=》告诉我们继承一个没有抽象方法的接口类,并不用重写也能实例化。毕竟没有抽象方法。

那有抽象方法的接口类呢?

1.继承有抽象方法的接口类但没有重写抽象方法=》报错

from abc import ABCMeta, abstractmethod

class Drawable(metaclass=ABCMeta):

    @abstractmethod
def size(self):
return 'Drawable size' # @abstractmethod
def draw(self, x, y, scale=1.0):
pass def double_draw(self, x, y):
self.draw(x, y, scale=2.0) class Cicle(Drawable):=====》继承但没有实现抽象方法 def paint(self):
pass c=Cicle()

TypeError: Can't instantiate abstract class Cicle with abstract methods size

2,继承有抽象方法的接口类必须重写抽象方法

from abc import ABCMeta, abstractmethod

class Drawable(metaclass=ABCMeta):

    @abstractmethod
def size(self):
return 'Drawable size' # @abstractmethod
def draw(self, x, y, scale=1.0):
pass def double_draw(self, x, y):
self.draw(x, y, scale=2.0) class Cicle(Drawable):
def size(self):
return 'Cicle size' def paint(self):
pass c=Cicle()

成功了

3.继承有抽象方法的非接口类呢?

from abc import ABCMeta, abstractmethod

class Drawable:

    @abstractmethod
def size(self):
return 'Drawable size' # @abstractmethod
def draw(self, x, y, scale=1.0):
pass def double_draw(self, x, y):
self.draw(x, y, scale=2.0) class Cicle(Drawable): def paint(self):
pass c=Cicle()

成功====》单独存在抽象方法并没啥用

进阶:

1.ABC是python 3.4之后新增的类,之前必须使用metaclass=ABCMeta,现在多了一个简单的方法ABC

2.abc模块还定义了'abstractclassmethod', 'abstractproperty', 'abstractstaticmethod'三个装饰器,但是因为可以通过@abstractmethod上堆叠实现,就显得多余二废弃了。

from abc import *

class A(ABC):

    @property
@abstractmethod
def func(self):
pass @abstractproperty
def func2(self):
pass @classmethod
@abstractmethod
def func3(cls):
pass @abstractclassmethod
def func4(cls):
pass @staticmethod
@abstractmethod
def func5():
pass @abstractstaticmethod
def func6():
pass

虚拟子类

注册虚拟子类的方式在抽象基类上调用register方法

注册的类就会变成抽象基类的虚拟子类,而且issubclass和isinstance等函数都能识别,但是注册的类并不会从抽象基类中继承任何方法。

虚拟子类不会继承注册的抽象基类,而且任何时候都不会检查子类是否符合抽象基类的接口(即使在实例化的时候也不会检查)。

作为抽象类的虚拟子类无需实现其需要的抽象方法
from abc import *
class A(metaclass=ABCMeta):
@abstractmethod
def func1(self):
pass @A.register
class B:
def func2(self):
pass class C(A):
def func2(self):
pass b=B() #===》作为抽象类的虚拟子类无需实现其需要的抽象方法
# c=C()==>继承抽象类没有实现抽象方法报错
print("OK")

类的继承关系在一个特殊属性中制定--__mro__即方法解析顺序,这个属性会按顺序列出所有 ‘真实的’ 类和其超类。

from abc import *
class A(ABC):
@abstractmethod
def func1(self):
pass class B:
def func1(self):
pass @A.register
class C(B):
def func2(self):
pass b=B()
c=C()
print(issubclass(C,A))
print(isinstance(c,A))
print(isinstance(c,B))
# True
# True
# True print(C.__mro__)
# (<class '__main__.C'>, <class '__main__.B'>, <class 'object'>)
# 没有A类

python之抽象类&abc模块+虚拟子类&register的更多相关文章

  1. python abc模块

    面向对象的设计中,抽象类,接口这些必不可少的东西,在python中是如何提现的呢? python作为一个动态语言,没有强类型的检查,而是以鸭子类型的方式提现,在执行的时候python不严格要求你必须是 ...

  2. Python装饰器、metaclass、abc模块学习笔记

    (博客原创作品,转载请注明出处!) 最近接触到了Python中的decorator,metaclass,abc Module,six.add_metaclass等内容,这里做一个简单的笔记. 主要资源 ...

  3. 第7.21节 Python抽象类—register注册虚拟子类

    上两节介绍了Python抽象类的真实子类的定义和使用,本节介绍另一种抽象类的实现方法:虚拟子类方法. 一.    相关概念 虚拟子类是将其他的不是从抽象基类派生的类"注册"到抽象基 ...

  4. 第7.19节 Python中的抽象类详解:abstractmethod、abc与真实子类

    第7.19节 Python中的抽象类详解:abstractmethod.abc与真实子类 一.    引言 前面相关的章节已经介绍过,Python中定义某种类型是以实现了该类型对应的协议为标准的,而不 ...

  5. Python抽象类(abc模块)

    1.抽象类概念 抽象类是一个特殊的类,只能被继承,不能实例化 2.为什么要有抽象类 其实在未接触抽象类概念时,我们可以构造香蕉.苹果.梨之类的类,然后让它们继承水果这个基类,水果的基类包含一个eat函 ...

  6. 学习Python的ABC模块(转)

    http://yansu.org/2013/06/09/learn-Python-abc-module.html 1.abc模块作用 Python本身不提供抽象类和接口机制,要想实现抽象类,可以借助a ...

  7. 001学习Python的ABC模块(转)

    http://yansu.org/2013/06/09/learn-Python-abc-module.html 1.abc模块作用 Python本身不提供抽象类和接口机制,要想实现抽象类,可以借助a ...

  8. Python基础-接口与归一化设计、抽象类、继承顺序、子类调用父类,多态与多态性

    一.接口与归一化设计 Java接口是一系列方法的声明,是一些方法特征的集合,一个接口只有方法的特征没有方法的实现,因此这些方法可以在不同的地方被不同的类实现,而这些实现可以具有不同的行为(功能). 由 ...

  9. python基础——抽象类

    python基础--抽象类 1  什么是抽象类 与java一样,python也有抽象类的概念但是同样需要借助模块实现,抽象类是一个特殊的类,它的特殊之处在于只能被继承,不能被实例化 2 为什么要有抽象 ...

随机推荐

  1. 学习笔记:CentOS7学习之二十三: 跳出循环-shift参数左移-函数的使用

    目录 学习笔记:CentOS7学习之二十三: 跳出循环-shift参数左移-函数的使用 23.1 跳出循环 23.1.1 break和continue 23.2 Shift参数左移指令 23.3 函数 ...

  2. axios 使用post方式传递参数,后端接收不到

    最近做vue项目,做图片上传的功能,使用get给后台发送数据,后台能收到,使用post给后台发送图片信息的时候,vue axios post请求发送图片base64编码给后台报错HTTP 错误 414 ...

  3. nginx+uwsgi02---django部署(不推荐)

    1.文件结构 myweb/ ├── manage.py ├── myweb/ │ ├── __init__.py │ ├── settings.py │ ├── urls.py │ └── wsgi. ...

  4. HashMap集合排序方法

    首先我们先来看看Map集合获取元素的三种常见方法(1)entrySet(),(2)keySet(),(3)values() 1. entrySet():(1)先返回map集合的所有"映射&q ...

  5. java中讲讲DataInputStream的用法,举例?

    [学习笔记] 2.4 DataInputStream的用法 马 克-to-win:DataInputStream顾名思义:就是专门用来读各种各样的数据的,比如(int,char,long等),一定要注 ...

  6. 复合模式MVC

    这里也只说一下简单的原理. Model:模型实现处理数据的切逻辑. View:视图呈现模型的数据和状态. Control:解读视图对模型的操作. 视图和模型之间使用观察者模式,只要模型的状态改变视图立 ...

  7. IPv4

    1.IPv4分类地址 PC0(192.168.0.1) 和PC1(172.168.0.1)如果要ping通的话需要设置各自网关 PC0  设置IP  和  默认网关=路由器设置IP 2.Gigabit ...

  8. python — mysql基础知识

    目录 1 . 数据库的介绍 2. mysql 1 . 数据库的介绍 1.为什么要用数据库? 很多功能如果只是通过操作文件来改变数据是非常繁琐的,程序员需要做很多事情 对于多台机器或者多个进程操作用一份 ...

  9. 把.exe的格式的运行程序加到电脑本地服务的办法(本文来源于百度)

    Instsrv.exe(可以给系统安装和删除服务) Srvany.exe(可以让程序以服务的方式运行) 方法/步骤     要实现这个功能要用到微软提供的两个小工具“instsrv.exe”和“srv ...

  10. 常用javascript内置对象——String对象

    创建 String 对象的语法: 1:new String(s); :2:String(s); :3:直接赋值 String中属性 String中方法 <script> window.on ...