前言

”面向接口编程“写 Java 的朋友耳朵已经可以听出干茧了吧,当然这个思想在 Java 中非常重要,甚至几乎所有的编程语言都需要,毕竟程序具有良好的扩展性、维护性谁都不能拒绝。

最近无意间看到了我刚开始写 Python 时的部分代码,当时实现的需求有个很明显的特点:

不同对象具有公共的行为能力,但具体每个对象的实现方式又各不相同。

说人话就是商户需要接入平台,接入的步骤相同,但具体实现不同。

作为一个”资深“ Javaer,需求还没看完我就洋洋洒洒的把各个实现类写好了:

当然最终也顺利实现需求,甚至把组里一个没写过 Java 的大哥唬的一愣一愣的,直呼牛逼。

不过事后也给我吐槽:

你这设计是不错,但是感觉好复杂,跟代码时要找到真正的业务逻辑(实现类)得绕几圈。

截止目前 Python 写多了,我总算是能总结他的感受:就是不够 Pythonic。

虽说 Python 没有类似 Java 这样的 Interface 特性,但作为面向对象的高级语言也是支持继承的;

在这里我们也可以利用继承的特性来实现面向接口编程:

class Car:

def run(self):

pass

class Benz(Car):

def run(self):

print("benz run")

class BMW(Car):

def run(self):

print("bwm run")

def run(car):

car.run()

if __name__ == "__main__":

benz = Benz()

bmw = BMW()

run(benz)

run(bmw)

代码非常简单,在 Python 中也没有类似于 Java 中的 extends 关键字,只需要在类声明末尾用括号包含基类即可。

这样在每个子类中就能单独实现业务逻辑,方便扩展和维护。

类型检查

由于 Python 作为一个动态类型语言,无法做到 Java 那样在编译期间校验一个类是否完全实现了某个接口的所有方法。

为此 Python 提供了解决办法,那就是 abc(Abstract Base Classes) ,当我们将基类用 abc 声明时就能近似做到:

import abc

class Car(abc.ABC):

@abc.abstractmethod

def run(self):

pass

class Benz(Car):

def run(self):

print("benz run")

class BMW(Car):

pass

def run(car):

car.run()

if __name__ == "__main__":

benz = Benz()

bmw = BMW()

run(benz)

run(bmw)

一旦有类没有实现方法时,运行期间便会抛出异常:

bmw = BMW()

TypeError: Can't instantiate abstract class BMW with abstract methods run

虽然无法做到在运行之前(毕竟不需要编译)进行校验,但有总比没有好。

鸭子类型

以上两种方式看似已经毕竟优雅的实现面向接口编程了,但实际上也不够 Pythonic。

在继续之前我们先聊聊接口的本质到底是什么?

在 Java 这类静态语言中面向接口编程是比较麻烦的,也就是我们常说的子类向父类转型,因此需要编写额外的代码。

带来的好处也是显而易见,只需要父类便可运行。

但我们也不必过于执着于接口,它本身只是一个协议、规范,并不特指 Java 中的 Interface,甚至有些语言压根没有这个关键字。

动态语言的特性也不需要强制校验是否实现了方法。

在 Python 中我们可以利用鸭子类型来优雅的实现面向接口编程。

在这之前先了解下鸭子类型,借用维基百科的说法:

“当看到一只鸟走起来像鸭子、游泳起来像鸭子、叫起来也像鸭子,那么这只鸟就可以被称为鸭子。”

我用大白话翻译下就是:

即便两个完全不想干的类,如果他们都实现了相同的方法,那就可以把他们当做同一类型的类来使用。

举个简单例子:

class Order:

def create(self):

pass

class User:

def create(self):

pass

def create(obj):

obj.create()

if __name__ == "__main__":

order = Order()

user = User()

create(order)

create(user)

这里的 order 和 user 本身完全没有关系,只是他们都有相同方法,又得益于动态语言没法校验类型的特点,所以完全可以在运行的时候认为他们是同一种类型。

因此基于鸭子类型,之前的代码我们可以稍作简化:

class Car:

def run(self):

pass

class Benz:

def run(self):

print("benz run")

class BMW:

def run(self):

print("bwm run")

def run(car):

car.run()

if __name__ == "__main__":

benz = Benz()

bmw = BMW()

run(benz)

run(bmw)

因为在鸭子类型中我们在意的是它的行为,而不是他们的类型;所以完全可以不用继承便可以实现面向接口编程。

总结

我觉得平时没有接触过动态类型语言的朋友,在了解完这些之后会发现新大陆,就像是 Python 老手第一次使用 Java 时;虽然觉得语法啰嗦,但也会羡慕它的类型检查、参数验证这类特点。

动静语言之争这里不做讨论了,各有各的好,鞋好不好穿只有自己知道。

随便提一下其实不止动态语言具备鸭子类型,有些静态语言也能玩这个骚操作,感兴趣下次再介绍。

python 接口编程_Python 中的面向接口编程相关推荐

  1. python面向接口编程_Python 中的面向接口编程

    前言 "面向接口编程"写 Java 的朋友耳朵已经可以听出干茧了吧,当然这个思想在 Java 中非常重要,甚至几乎所有的编程语言都需要,毕竟程序具有良好的扩展性.维护性谁都不能拒绝 ...

  2. python接口编程_Python 中的面向接口编程

    前言 "面向接口编程"写 Java 的朋友耳朵已经可以听出干茧了吧,当然这个思想在 Java 中非常重要,甚至几乎所有的编程语言都需要,毕竟程序具有良好的扩展性.维护性谁都不能拒绝 ...

  3. python 编程接口_Python 中的面向接口编程

    前言 "面向接口编程"写 Java 的朋友耳朵已经可以听出干茧了吧,当然这个思想在 Java 中非常重要,甚至几乎所有的编程语言都需要,毕竟程序具有良好的扩展性.维护性谁都不能拒绝 ...

  4. spring中AOP(面向切面编程)

    spring中AOP(面向切面编程) 面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术.AOP是OOP的延续,是软件开发中的一个热点,也是spring框架中的一个重要内容 ...

  5. Java 中的面向数据编程

    近年来, Amber项目为 Java 带来了许多新特性-- 局部变量类型推断. 文本块. 记录类. 封印类. 模式匹配 等等.虽然这些特性都是独立的,但也可以组合在一起使用.具体地说,记录类.封印类和 ...

  6. 神理论 | 我面向淘宝编程,而你面向什么编程?(文末赠书5本)

    神理论 | 我面向淘宝编程,而你面向什么编程? 陌生人一定会疑问,嵌入式跟taobao有半毛钱关系啊?那么本文,博主就跟你聊聊,如何面向淘宝学习嵌入式! 文章目录 1 写在前面 2 面向XXX编程 3 ...

  7. Java中的面向接口编程

    面向接口编程是很多软件架构设计理论都倡导的编程方式,学习Java自然少不了这一部分,下面是我在学习过程中整理出来的关于如何在Java中实现面向接口编程的知识.分享出来,有不对之处还请大家指正. 接口体 ...

  8. java的知识点13——多态、对象的转型(casting)、final关键字、抽象方法和抽象类、接口的作用、如何定义和使用接口?、接口的多继承、面向接口编程

    多态 多态指的是同一个方法调用,由于对象不同可能会有不同的行为.现实生活中,同一个方法,具体实现会完全不同. 多态的要点: 1. 多态是方法的多态,不是属性的多态(多态与属性无关). 2. 多态的存在 ...

  9. AOP编程过程中的JoinPoint接口

    在AOP编程过程中需要使用到的一个 JoinPoint 接口,用来获取调用的信息,下面是这个接口的代码 // // Source code recreated from a .class file b ...

最新文章

  1. 汇编语言 循环让字符串中的字母变成大写
  2. 六面!终斩腾讯NLP暑期实习offer
  3. 十三种基于直方图的图像全局二值化算法原理、实现、代码及效果。
  4. 【总结】DIV+CSS有可能遇到的问题
  5. 对抗训练浅谈:意义、方法和思考(附Keras实现)
  6. python顺时针旋转_Python之二维数组N*N顺时针旋转90度
  7. java虚拟机的生命周期_深入理解Java虚拟机——JVM的生命周期
  8. dubbo+rabbitmq+hystrix实现服务的高可用
  9. [必看]首先要求做到的事情![sumtec]
  10. r语言插补法_R语言用多重插补法估算相对风险
  11. CentOS 7添加应用快捷方式到桌面
  12. 基于CUDA的GPU优化建议
  13. 华为三层交换机-路由-硬件防火墙的配置
  14. 教你如何打造网页爬虫工具(实现思路及源码下载)
  15. simulink 调整系统采样时间 示波器显示时间
  16. 折叠面板(Collapse)
  17. STM32读写ADXL345 中断功能
  18. EricChu的雅思之路——2020雅思A类考试切身体会\自学备考建议\考试完整流程(最详细版请耐心阅读)
  19. svn提交代码报错:svn: E175002: Unexpected HTTP status 502 ‘Bad Gateway‘
  20. 流落在帝都的 80 后北漂们的未来在哪儿?

热门文章

  1. boost::serialization模块测试extended_type_info的实现,使用多个共享库时有效
  2. boost::proto::protect相关的测试程序
  3. boost::multi_index模块相关的测试程序
  4. boost::geometry::dot_product用法的测试程序
  5. GDCM:提取DICOM文件的加密内容到der文件的测试程序
  6. GDCM:提取JP2文件所有解析度的测试程序
  7. Boost:是否支持sse2指令的测试程序
  8. Boost:bind绑定__cdecl(成员函数)测试程序
  9. ITK:将数组转换为图像
  10. ITK:在一张图像中设置像素值