如何使用Python基于接口编程的方法实现

目录
  • 先通过一个实例来了解下接口到底解决什么问题。
  • 定义一个接口
  • 定义类,继承接口
  • Python 抽象基类的介绍 (PEP3119)

软件行业,唯一不变的就是变化。产品经理会变,产品需求会变,代码同样要跟着变。

不同的代码设计,变化所带来的工作量更是不同,有的每改一次需求,近乎一次重构,而有的只需要修改一个配置文件,或者在类里添加一行代码。当然比较好的代码设计,由于有着良好的可扩展性,高内聚,低耦合,因而易维护, 以少变应万变。如果才能有好的代码设计,就需要我们学习设计模式。今天为你分享的是在Python中,如何基于接口编程。

1994 年 GoF 的《设计模式》一书中有一个重要的原则就是:基于接口而非实现编程,英文源文是「Program to an interface,not an implementaion」,这里的所说的 interface,并不是特定编程语言中的接口,它是语言无关的,是指开发者提供给使用者的一个功能列表,理解了这一点非常重要。接口在 java 语言中是有关键字 interface 来实现的,java 不支持类的多重继承,但支持接口的多重继承,所在 java 开发者对接口非常熟悉了,Python 其实完全不需要 Java 那样的设计,但可以借鉴接口的优点。

先通过一个实例来了解下接口到底解决什么问题。

比如你正在实现一个图片上传功能,目前采用七牛云来存储,你的类可能是这样的。

class QnyImageStore(object):

    def getToken():
        pass

    def upload_qny(self,image):
        print("Qny upload.")
        #do something

    def download_qny(self,image):
        print("Qny download.")
        #do something

实际的开发中,代码会有很多行,函数也不止三个,它被成百上千个地方被调用,分散在好几百个文件中。 过了一段时间,公司自建了私有云,要求不能再使用七牛云了,要改成自己的云存储,于是你不得不重新写一个类:

class OwnImageStore(object):

    def upload_own(self,image):
        print("Qny upload.")
        #do something

    def download_own(self,image):
        print("Qny download.")
        #do something

然后你在使用七牛去的地方,都进行替换,还要替换函数的名称,最后还要多次测试,生活哪一处没有考虑到,运行报错。好不容易改好了,突然有一天,需求又变了,由于自己的云服务经常出问题,于是要换阿里云。经过上次的一翻痛苦折腾,看到这次又要改,直接吐血。

其实问题就在于你写的代码不够通用,命名不够抽象。假如你的类的函数命名使用 upload,download 这样,那么你修改的代码量可能会减少到一半,只替换一下类名就可以了。实际上,我们可以使用接口来减少代码的改动量:通过接口和实现相分离的模式,封装不稳定的实现,暴露稳定的接口。上下游系统在使用我们开发的功能时,只需要使用接口中声明的函数列表,这样当实现发生变化的时候,上游系统的代码基本上不需要做改动,以此来降低耦合性,提高扩展性。下面就该问题,提供一种基于接口的代码实现方式。

定义一个接口

from abc import ABCMeta, abstractmethod

class ImageStore(metaclass = ABCMeta):

    @abstractmethod
    def upload(self,image):
        pass
        #raise NotImplementedError

    @abstractmethod
    def download(self,image):
        pass
        #raise NotImplementedError

定义了该接口之后,任何继承该接口的类要想正确的使用,必须重写 upload 和 download 方法,否则均会抛出异常,这里我们不需要自己在接口中抛出异常,标准库 abc 已经为我们做好了这些工作。

定义类,继承接口

目的其实是是为了强制约束,也就是说必须实现 upload 和 download 方法,在编译时进行检查,确保程序的健壮。

class OwnImageStore2(ImageStore):

    def upload(self,image):
        print("Own upload.")
        #do something

    def download(self,image):
        print("Own download.")
        #do something

class QnyImageStore2(ImageStore):

    def getToken():
        pass

    def upload(self,image):
        print("Qny upload.")

    def download(self,image):
        print("Qny download.")
        #do something

接下来,我们定义一个接口,可以自动的根据类型来选择调用对应对象的方法。

class UsedStore(object):

    def __init__(self, store):
        if not isinstance(store, ImageStore): raise Exception('Bad interface')
        self._store = store

    def upload(self):
        self._store.upload('image')

    def download(self):
        self._store.download('image')

最后,我们可以在配置文件中指明我们使用的是哪个具体的接口:

#在其他文件中,应该这样调用
img = QnyImageStore2()
# img = OwnImageStore2() 把这些放在配置文件中,只需要更新配置文件就可以替换
store = UsedStore(img)
store.upload()
store.download()

这样,后面再增加新的图片存储,我们只需要添加相应的类,继承接口,并修改下配置文件即可,减轻大量的代码修改工作。

Python 抽象基类的介绍 (PEP3119)

python 标准库 abc,全称是Abstract Base Classes,它提供以下功能:

  • 一种重载isinstance()和issubclass()的方法
  • 一个新的模块abc作为“Abstract Base Classes支持框架”。它定义了一个用于abc的元类和一个可以用来定义抽象方法的装饰器
  • 容器和迭代器的特定抽象基类,将被添加到 collections 模块

基本原理:

在面向对象程序设计领域,与对象交互的设计模式可以分为两个基本类别,即“调用”和“检查”。

调用是指通过调用对象的方法与对象进行交互。 通常,这会与多态性结合使用,因此调用给定方法可能会根据对象的类型运行不同的代码。

检查是指外部代码(在对象的方法之外)检查该对象的类型或属性,并根据该信息来决定如何处理该对象的能力。

两种使用模式均服务于相同的通用目的,即能够以统一的方式支持处理多种多样且可能新颖的对象,但同时允许为每种不同类型的对象定制处理决策。

在经典的 OOP 理论中,调用是首选的设计模式,并且不鼓励检查,因为检查被认为是较早的过程编程风格的产物。 但是,实际上,这种观点过于教条和僵化,导致了某种设计僵化,与诸如 Python 之类的语言的动态特性大相径庭。

特别是,通常需要以对象类的创建者无法预期的方式处理对象。 内置到满足该对象的每个可能用户需求的每个对象方法中,并非总是最佳的解决方案。 而且,有许多强大的调度哲学与严格地封装在对象中的经典OOP行为要求形成鲜明对比,例如规则或模式匹配驱动的逻辑

另一方面,经典的 OOP 理论家对检查的批评之一是缺乏形式主义和被检查内容的特殊性质。 在诸如 Python 这样的语言中,几乎可以通过外部代码反映并直接访问对象的任何方面,有很多不同的方法来测试对象是否符合特定的协议。 例如,如果询问“此对象是否是可变序列容器?”,则可以寻找“列表”的基类,或者可以寻找名为“ getitem”的方法。 但是请注意,尽管这些测试看似显而易见,但它们都不正确,因为其中一个会产生假阴性,而另一个会产生假阳性。

普遍同意的补救措施是对测试进行标准化,并将其分组为正式形式。 通过继承机制或其他某种方式,通过与每个类关联一组标准的可测试属性,最容易做到这一点。 每个测试都带有一组承诺:它包含有关类的一般行为的承诺,以及有关其他可用的类方法的承诺。

PEP为组织这些测试提出了一种特殊的策略,称为抽象基类(ABC)。 ABC只是添加到对象的继承树中的Python类,以将对象的某些功能发送给外部检查器。 使用isinstance()完成测试,并且特定ABC的存在意味着测试已通过。

此外,ABC定义了建立该类型特征行为的最少方法集。 根据对象的ABC类型区分对象的代码可以相信,这些方法将始终存在。 这些方法中的每一个都附带有ABC文档中描述的广义抽象语义定义。 这些标准的语义定义不是强制性的,但强烈建议使用。

像Python中的所有其他内容一样,这些承诺属于绅士协议的性质,在这种情况下,这意味着尽管该语言确实执行了ABC中做出的某些承诺,但具体类的实现者必须确保 剩下的保留下来。

看完上面的描述,你可以简单的理解为,ABC 是一个基类,继承它,你可以写一个类似于 java 的接口,接口中的方法将始终存在,可以放心使用,不需要再进行探测。

PEP3119 还给了样例代码让你理解:

from abc import ABCMeta, abstractmethod

class A(metaclass=ABCMeta):
    @abstractmethod
    def foo(self): pass

A()  # raises TypeError

class B(A):
    pass

B()  # raises TypeError

class C(A):
    def foo(self): print(42)

C()  # works

多的不说了,希望你可以正确地使用 ABC,同时强烈推荐,学习 Python,就看 Python 的官方文档和 PEP 提案,这里有最权威的讲解。

此外,设置模式也是非常重要的编程之术和编程之道,它是基本功,基本功如果不够,把一台战斗机放你面前,你都不知道如何欣赏和品味。

掌握了设计模式,再看别人的代码,你会拥有火眼金睛,哪些是战斗机,哪些是拖拉机,对自己的学习和提升也非常有帮助,写的代码也会更加具有可维护性,可读性,可扩展性,灵活性。

到此这篇关于如何使用Python基于接口编程的方法实现的文章就介绍到这了,更多相关Python基于接口编程内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Python中的面向接口编程示例详解

    前言 "面向接口编程"写 Java 的朋友耳朵已经可以听出干茧了吧,当然这个思想在 Java 中非常重要,甚至几乎所有的编程语言都需要,毕竟程序具有良好的扩展性.维护性谁都不能拒绝. 最近无意间看到了我刚开始写 Python 时的部分代码,当时实现的需求有个很明显的特点: 不同对象具有公共的行为能力,但具体每个对象的实现方式又各不相同. 说人话就是商户需要接入平台,接入的步骤相同,但具体实现不同. 作为一个"资深" Javaer,需求还没看完我就洋洋洒洒的把各个实现

  • 如何使用Python基于接口编程的方法实现

    目录 先通过一个实例来了解下接口到底解决什么问题. 定义一个接口 定义类,继承接口 Python 抽象基类的介绍 (PEP3119) 软件行业,唯一不变的就是变化.产品经理会变,产品需求会变,代码同样要跟着变. 不同的代码设计,变化所带来的工作量更是不同,有的每改一次需求,近乎一次重构,而有的只需要修改一个配置文件,或者在类里添加一行代码.当然比较好的代码设计,由于有着良好的可扩展性,高内聚,低耦合,因而易维护, 以少变应万变.如果才能有好的代码设计,就需要我们学习设计模式.今天为你分享的是在P

  • 批处理与python代码混合编程的方法

    批处理可以很方便地和其它各种语言混合编程,除了好玩,还有相当的实用价值,比如windows版的ruby gem包管理器就是运用了批处理和ruby的混合编写,bathome出品的命令工具包管理器bcn 使用了bat+jscript的混编实现的. cn-dos和bathome论坛里先后有帖子介绍和示范了批处理和各种语言脚本的混合编程,有兴趣可以搜索看看. python不挑剔文件后缀,只要程序中包含正确的python代码都可以用python 解释器解释执行. 批处理与python的混合编程方法很简单,

  • Python基于SMTP发送邮件的方法

    在很多时候,使用 Python 发送邮件可能没有办法使用邮件服务器提供的 API,因为不是所有的邮件服务商都会提供 API 供客户使用的. 通常使用邮件 API 的邮件发送服务都需要额外的收费. 因此我们再邮件测试发送的时候,可能需要的是 SMTP 邮件发送服务,通常这个服务是所有邮件服务商都会提供的. 要使用 SMTP 邮件发送服务,你需要有下面的信息才可以完成和测试: SMTP 邮件服务器的地址,端口,登录用户名和登录用户密码 发送和接收邮件的地址 邮件的主题和正文 看起来是不是有点复杂,实

  • Python编程之基于概率论的分类方法:朴素贝叶斯

    概率论啊概率论,差不多忘完了. 基于概率论的分类方法:朴素贝叶斯 1. 概述 贝叶斯分类是一类分类算法的总称,这类算法均以贝叶斯定理为基础,故统称为贝叶斯分类.本章首先介绍贝叶斯分类算法的基础--贝叶斯定理.最后,我们通过实例来讨论贝叶斯分类的中最简单的一种: 朴素贝叶斯分类. 2. 贝叶斯理论 & 条件概率 2.1 贝叶斯理论 我们现在有一个数据集,它由两类数据组成,数据分布如下图所示: 我们现在用 p1(x,y) 表示数据点 (x,y) 属于类别 1(图中用圆点表示的类别)的概率,用 p2(

  • 基于Python的接口自动化读写excel文件的方法

    引言 使用python进行接口测试时常常需要接口用例测试数据.断言接口功能.验证接口响应状态等,如果大量的接口测试用例脚本都将接口测试用例数据写在脚本文件中,这样写出来整个接口测试用例脚本代码将看起来很冗余和难以清晰的阅读以及维护,试想如果所有的接口测试数据都写在代码中,接口参数或者测试数据需要修改,那不得每个代码文件都要一一改动?.因此,这种不高效的模式不是我们想要的.所以,在自动化测试中就有个重要的思想:测试数据和测试脚本分离,也就是测试脚本只有一份,其中需要输入数据的地方会用变量来代替,然

  • 谈一谈基于python的面向对象编程基础

    活在当下的程序员应该都听过"面向对象编程"一词,也经常有人问能不能用一句话解释下什么是"面向对象编程",我们先来看看比较正式的说法. 把一组数据结构和处理它们的方法组成对象(object),把相同行为的对象归纳为类(class),通过类的封装(encapsulation)隐藏内部细节,通过继承(inheritance)实现类的特化(specialization)和泛化(generalization),通过多态(polymorphism)实现基于对象类型的动态分派.

  • Python基于time模块求程序运行时间的方法

    本文实例讲述了Python基于time模块求程序运行时间的方法.分享给大家供大家参考,具体如下: 要记录程序的运行时间可以利用Unix系统中,1970.1.1到现在的时间的毫秒数,这个时间戳轻松完成. 方法是程序开始的时候取一次存入一个变量,在程序结束之后取一次再存入一个变量,与程序开始的时间戳相减则可以求出. Python中取这个时间戳的方法为引入time类之后,使用time.time();就能够拿出来.也就是Java中的System.currentTimeMillis(). 由于Python

  • Python实现数据库编程方法详解

    本文实例讲述了Python实现数据库编程方法.分享给大家供大家参考.具体分析如下: 用PYTHON语言进行数据库编程, 至少有六种方法可供采用. 我在实际项目中采用,不但功能强大,而且方便快捷.以下是我在工作和学习中经验总结. 方法一:使用DAO (Data Access Objects) 这个第一种方法可能会比较过时啦.不过还是非常有用的. 假设你已经安装好了PYTHONWIN,现在开始跟我上路吧-- 找到工具栏上ToolsàCOM MakePy utilities,你会看到弹出一个Selec

  • python基于pyDes库实现des加密的方法

    本文实例讲述了python基于pyDes库实现des加密的方法.分享给大家供大家参考,具体如下: 下载及简介地址:https://twhiteman.netfirms.com/des.html 如需要在python中使用des加密,可以直接使用pyDes库加密,该库提供了CBC和ECB两种加密方式. 1.Windows下安装 下载后pyDes-x.x.x.zip并解压后,里面有setup.py文件,使用命令 setup.py --help可查看详细使用. 你可以使用命令python setup.

  • Python基于二分查找实现求整数平方根的方法

    本文实例讲述了Python基于二分查找实现求整数平方根的方法.分享给大家供大家参考,具体如下: x=int(raw_input('please input a int:')) if x<0: retrun -1 low=0 high=x ans=(low+high)/2.0 sign=ans while ans**2 !=x: if ans**2>x: high=ans else: low=ans ans=(low+high)/2.0 if sign==ans: break print ans

随机推荐