Python札记33_绑定方法和非绑定方法

在类中除了属性还有方法,通常情况下通过实例来进行调用类的方法。本篇札记讲解四个方面进行:

  • 绑定方法
  • 非绑定方法
  • 静态方法
  • 类方法

非绑定方法 unbound method

通过类的名字直接来调用方法对象(函数对象),叫做非绑定方法。其中Foo是类,bar是类中的方法。类名字---->方法对象

image.png
class Foo:
    def bar(self):  # 第一个参数必须是self
        print("this is a method of class")
f = Foo()   # 调用类
f.bar()     # 调用类中方法(类中的函数也称之为方法)
this is a method of class
  • 类中的函数也称之为方法
  • 当建立了实例f之后,当实例调用这个方法bar()的时候,Python解释器把实例已经作为第一个参数隐式地传给了该方法。
  • self就是一个实例!!!!不需要显式地写出self参数
# 实例显式地传给方法
Foo.bar(f)  
this is a method of class

Foo.bar()  # 通过类直接调用方法,不传递参数报错,缺少self
  • 实例化之后,self和实例f是一样的
  • 通常在类中使用self,在实例中使用f
  • 如果用类直接调用方法则会报错:
image.png

绑定方法 bound method

绑定方法指的是通过实例得到方法对象的过程。实例---->方法对象

描述器

在类的属性中有个__dict__的特殊属性,用来查看内部信息:

Foo.__dict__["bar"]   # bar 是函数对象
image.png
  • Python中有几个比较特殊的方法:__get__()、__set__()、__deleta__(),通常将具有这些方法的对象称之为描述器。描述器的使用如下:
    descr.get(self, obj, type=None)--->value
    descr.set(self, obj,value)--->None
    descr.delete(self, obj)--->None
  • 描述器的使用:
    Foo.dict['bar'].get(None, Foo)
image.png

结果:描述器的返回结果和Foo.bar是相同的。其中self指定为None,表示没有给定实例。

Foo.__dict__["bar"].__get__(f, Foo) # 给定实例f;与实例得到方法对象结果相同
image.png

类方法

lass Foo:   # 定义一个类
    lang = 'Java'  # 定义类的属性lang = "java"
    def __init__(self):  # 初始化方法__init__
        self.lang = 'python' # 实例属性,self是实例对象,属性数lang = "python"
        
def get_class_attr(cls):  
    return cls.lang   # 引用的对象具有lang属性
    
if __name__ == "__main__":
    print("Foo.lang(类属性值)", Foo.lang)  # 输出类lang属性值
    r = get_class_attr(Foo)   # 调用get_class_attr()方法,其中参数Foo刚好有lang属性
    print("get class attr:", r)  
    f = Foo()  # 创建实例f
    print("instance attribute(实例属性值)", f.lang)  # 参数实例的lang属性值
    

结果:

Foo.lang(类属性值) Java
get class attr: Java
instance attribute(实例属性值) python

缺陷:

  • get_class_attr()方法传入的参数不是随意的,必须具有lang属性
  • 此做法使得函数和类的耦合性太强,不利于后期维护,考虑将函数和类融为一体,直接将函数放在类里面。

使用装饰器来解决上面的问题

class Foo:   # 定义一个类
    lang = 'Java'  # 定义类的属性lang = "java"
    def __init__(self):  # 初始化方法__init__
        self.lang = 'python' # 实例属性,self是实例对象,属性数lang = "python"
        
    # 修改部分,加上装饰器,@classmethod装饰的函数第一个参数不是self
    @classmethod
    def get_class_attr(cls):  
        return cls.lang   # 引用的对象具有lang属性
    
if __name__ == "__main__":
    print("Foo.lang(类属性值)", Foo.lang)  # 输出类lang属性值
    r = get_class_attr(Foo)   # 调用get_class_attr()方法,其中参数Foo刚好有lang属性
    print("get class attr:", r)  
    f = Foo()  # 创建实例f
    print("instance attribute(实例属性值):", f.lang)  # 参数实例的lang属性值
    print("instance get_class_attr:", f.get_class_attr())
Foo.lang(类属性值) Java
get class attr: Java
instance attribute(实例属性值): python
instance get_class_attr: Java

结果:通过实例和类执行get_class_attr()方法得到的结果都是Java

  • 类方法:在类里面定义的方法,该方法由@classmethod所装饰第一个参数cls(约定俗称第一个参数为cls)的引用对象为类本身`。

静态方法

下面是一个有待优化的代码,因为类Foo里面使用了外面定义的函数select函数,不便于维护

import random

def select(n):
    a = random.randint(0, 100)    # 生成一个随机数
    return a - n > 0

class Foo:
    def __init__(self, name):     # 初始化函数
        self.name = name    # 实例self的 name属性为name
        
    def get_name(self, age):    # 第一参数必须是self
        if select(age) is True:     #   就是返回值大于0
            return self.name   
        else:
            return "this name is secret."
    
if __name__ == "__main__":
    f = Foo("Peter")
    name = f.get_name(27)
    print(name)

优化代码如下

import random

class Foo:   # 定义类,首字母大写 
    def __init__(self, name):  # 初始化函数
        self.name = name   # self实例的name属性为name
    
    def get_name(self, age):   # 将原来在外面的函数放在里面
        if self.select(age):   # 直接通过实例进行调用;也可以通过类进行调用:Foo.select()
            return self.name   # 返回实例self的name属性值
        else:
            return "the name is secret"
        
   # 增加装饰器 
    @staticmethod
    def select(n):   # 虽然在类里面,但是第一个参数不是self
        a = random.randint(1,100)
        return a - n > 0
    
if __name__ == "__main__":
    f = Foo("luolaoshi")
    name = f.get_name(22)
    print(name)

小结:

  • 将原来在类外面的函数放到里面
  • 移动到里面的方法的第一个参数不是self
  • 移到类的命名空间之内,前面必须加上@staticmethod进行装饰
  • 调用可以通过实例或者进行调用

下面对上面的四种方法进行一个小结:

  • 绑定方法:实例--->对象方法
  • 非绑定方法:类--->对象方法
  • 类方法:@classmethod,将方法写在类里面,该方法的第一个参数不是self,通常用cls。
  • 静态方法:@staticmethod,将方法移动到类里面,位于类的命名空间之内。第一个参数不是self;方法可以通过实例或类进行调用。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,076评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,658评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,732评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,493评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,591评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,598评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,601评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,348评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,797评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,114评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,278评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,953评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,585评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,202评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,180评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,139评论 2 352