你知道什么是Python里的鸭子类型和猴子补丁吗?

疯狂软件李刚

共 2079字,需浏览 5分钟

 ·

2020-06-25 23:24

有时候我们会听到Python里所谓的鸭子类型猴子补丁的说法,乍一听还以为是来到了动物园,Python这只大蟒蛇还可以和鸭子和猴子一起玩耍?d3ecae42a4b7f7f3edfe3d191eabd239.webpd3ecae42a4b7f7f3edfe3d191eabd239.webpd3ecae42a4b7f7f3edfe3d191eabd239.webp


非也非也,鸭子类型和猴子补丁实际上是两个生动有趣的比喻,用来说明Python的动态特性。


鸭子类型?(duck type) 是对Python中数据类型本质上是由属性和行为来定义的一种解读。


猴子补丁?(monkey patching)是对Python中类和模块可以在外部被动态修改这种特性的一个比喻。


让我们来一探究竟吧!


一,鸭子类型

鸭子类型是对Python中数据类型本质上是由属性和行为来定义的一种解读。


Python是一种动态语言,不像Java和C++这种强类型语言,Python里实际上没有严格的类型检查。


只要某个对象具有鸭子的方法,可以像鸭子那样走路和嘎嘎叫,那么它就可以被其它函数当做鸭子一样调用。


# 这是一个鸭子(Duck)类
class Duck:
    def __init__(self,name):
        self.name = name
    def swim(self):
        print("A duck named " + self.name+" is swimming...")
    def call(self):
        print("gay...gay...gay...")

# 这是一个鹅(Goose)类
class Goose:
    def __init__(self,name):
        self.name = name
    def swim(self):
        print("A goose named " + self.name+" is swimming...")
    def call(self):
        print("goo...goo...goo...")

# duckshow这个函数设计的本意是需要传入一个Duck对象作为参数的。 
def duckshow(duck):
    duck.swim()
    duck.call()

yaya = Duck("yaya")
ee = Goose("ee")

# 但由于Python是动态语言,其数据类型属于鸭子类型,没有严格类型检查,
# Goose这个类具有和Duck这个类相同的方法,
# ee这只鹅划起水来像只鸭子,叫起来也像一只鸭子,
# 所以duckshow这个函数也可以对ee进行作用。

duckshow(yaya)
duckshow(ee)

输出结果如下:
A duck named yaya is swimming...gay ... gay ... gay ...A goose named ee is swimming...goo ... goo ... goo ...

二,猴子补丁

猴子补丁是对Python中模块和类可以在外部被动态修改这种特性的一个比喻。


为什么叫做猴子补丁呢?在模块和类的外部对模块和类进行修改是一种非常耍赖的做法,会破坏代码的封装结构,这种事情大概只有淘气的猴子喜欢去做,因此形象地称之为猴子补丁。


# 定义一个Dog类
class Dog:
    def __init__(self,name,age):
        self.name = name
        self.age = age

    def sleep(self):
        print("Zzz...Zzz..Zzz...")

# 在类的外部给Dog这个类添加猴子补丁
def speak(self):
    print("I think myself a hero and very handsome!")

Dog.speak = speak
Dog.home = "Earth"

# 与类的内部定义的属性和方法无差异
snoopy = Dog("snoopy",3)
snoopy.sleep()
snoopy.speak()
print(snoopy.home)


输出结果如下:

Zzz ... Zzz ... Zzz ...I think myself a hero and very handsome!Earth


对于库中已经定义好的类,给它们添加猴子补丁有时候会非常方便,相当于可以自由地在外面做扩展,又不用修改源文件。


import numpy as np
import pandas as pd

df = pd.DataFrame(np.random.randint(1,10,size=(5,5)),
                  columns = list("abcde"))
#df.info()
#DataFrame自带方法info打印信息太多,展示不直观。
#给DataFrame定义一个猴子补丁方法memory简洁展示其内存消耗。

def memory(self):
    mem = self.__sizeof__()
    if mem<1024:
        return("%s B"%mem)
    elif mem<1024*1024:
        return("%s KB"%mem/1024)
    elif mem<1024**3:
        return("%s MB"%mem/1024**2)
    else:
        return("%s GB"%mem/1024**3

pd.DataFrame.memory = memory

print(df.memory())


输出结果如下:

280 B


这就是Python里的鸭子类型和猴子补丁的一个简单介绍,你耍明白了吗???????


浏览 15
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报