[转载]Python方法绑定——Unbound/Bound method object的一些梳理

本篇主要总结Python中绑定方法对象(Bound method object)和未绑定方法对象(Unboud method object)的区别和联系。主要目的是分清楚这两个极容易混淆的概念,顺便将Python的静态方法,类方法及实例方法加以说明

OK,下面开始

1. 一个方法引发的“血案”

类中所定义的函数称为方法举例:

>>>class Foo(object):
...        def  foo():
...            print ‘call foo‘

然后令人困惑的地方就来了:当你尝试使用类名.方法名调用函数foo时,会出现如下错误

>>> Foo.foo()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: unbound method foo() must be called with Foo instance as first argument (got nothing instead)

看一下报错信息发现需要一个Foo的实例(instance)来调用,OK,于是调用如下:

>>> Foo().foo()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: foo() takes no arguments (1 given)

-.-!!!
估计脾气不好的看到做到这里想要骂街了。
因为从字面上看Foo( ).foo( )并没有传递任何参数,而报错信息却显示(1 given)。

在Python中一切皆对象,方法是函数,所以我们来仔细查看一下函数对象foo

>>> Foo.foo
<unbound method Foo.foo>
>>> Foo().foo
<bound method Foo.foo of <__main__.Foo object at 0x7ff33b424d50>>

咦~,发现一个有趣的现象:
通过类名Foo获取类函数属性foo时,得到的是unbound method object,通过实例Foo()获取类的函数属性foo时,得到的是bound method object。
在来看看这两个对象的类型:

>>> type(Foo.foo)
<type ‘instancemethod‘>
>>> type(Foo().foo)
<type ‘instancemethod‘>

于是我们产生了更大的疑问:为什么同样是实例方法(instancemethod),获取方式的不同,会导致获得不同的对象呢?

2. bound/unbound method是怎么来的

下面让我们来一层层揭开这个bound/unbound method的面纱。首先,我们知道,对于类,其属性是存放在__dict__字典中,即:

>>> Foo.__dict__
dict_proxy({‘__dict__‘: <attribute ‘__dict__‘ of ‘Foo‘ objects>, ‘__module__‘: ‘__main__‘, ‘foo‘: <function foo at 0x7ff33b42a5f0>, ‘__weakref__‘: <attribute ‘__weakref__‘ of ‘Foo‘ objects>, ‘__doc__‘: None})

在其中我们看到了‘foo‘: <function foo at 0x7ff33b42a5f0>。然后利用字典查看foo:

>>> Foo.__dict__[‘foo‘]
<function foo at 0x7ff33b42a5f0>

可以看到foo是一个函数对象,根据上一小节最后一个例子的信息,我们发现,foo是有绑定行为的。

在Python中使用描述器(有翻译的链接)来表示具有“绑定”行为的对象属性,使用描述器协议方法来控制对具有绑定行为属性的访问,这些描述器协议方法包括:__get__()、__set__()和__delete__()。
根据上面这段难以让人理解的描述,我们可以大胆的猜测,Foo的属性foo是一个描述器,它通过__get__()方法来控制对foo的访问。
根据描述器协议方法descr.__get__(self, obj, type=None) --> value,我们尝试如下:

>>> Foo.__dict__[‘foo‘].__get__(None,Foo)
<unbound method Foo.foo>

于是,我们惊讶的看到这个结果竟然与上一小节看到的结果相同!
这绝不是偶然。
事实上,根据官方文档的描述,调用Foo.foo时,Python会根据查找链从Foo.__dict__[‘foo‘]开始,然后查找type(Foo).__dict__[‘foo‘],一路向上查找type(Foo)的所有基类。Foo.foo会被转换为Foo.__dict__[‘foo‘].__get__(None,Foo)。
也就是说,我们在代码中使用Foo.foo实际上会被转化成
Foo.__dict__[‘foo‘].__get__(None,Foo)
对于根据描述器协议方法descr.__get__(self, obj, type=None) --> value的参数列表,由于其self参数在这里被赋予了None,所以没有给定实例,因此认为是未绑定(unbound)
(当然这是一种便于理解的描述,其根本机制请移步这里)
那么一个很简单的推理就是:如果self参数给定了实例对象,那么,得到的就是bound method,如下。

>>> Foo.__dict__[‘foo‘].__get__(Foo(),Foo)
<bound method Foo.foo of <__main__.Foo object at 0x7ff33b424d50>>

因此,可以有如下理解:

当通过类来获取函数属性的时候,得到的是非绑定方法对象当通过实例来获取函数属性的时候,得到的是绑定方法对象

3. methods, static method and class method

如果有使用Python方法的经验,那么一定注意过self的使用,请看下面这个例子:

>>> class Foo(object):
...     def foo():
...             print ‘call foo‘
...     def foo_one(self):
...             print ‘call foo_one‘
...
>>> Foo.foo()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: unbound method foo() must be called with Foo instance as first argument (got nothing instead)
>>> Foo().foo()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: foo() takes no arguments (1 given)
>>> Foo.foo_one()
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: unbound method foo_one() must be called with Foo instance as first argument (got nothing instead)
>>> Foo().foo_one()
call foo_one

这个例子定义了两个method:foo()和foo_one(self)。
可以看到,同样使用类名.方法名()调用时,所报的错误相同。但是在使用实例名.方法名()调用时,foo_one是可以调用成功的。
为什么呢?
原因在于当使用Foo().foo_one()调用时,Python做了如下修改:

>>> Foo.foo_one(Foo())
call foo_one

将实例Foo()作为第一个参数传递进去,因此,函数foo_one(self)调用成功。这也解释了为什么Foo().foo()调用不成功。因为foo的定义为foo(),当调用Foo().foo()时,Python做了如下修改:

>>> Foo.foo(Foo())
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
TypeError: foo() takes no arguments (1 given)

传入了一个参数Foo(),所以会出现foo() takes no arguments (1 given)的错误。
我曾经看到有的人把foo()这种参数列表中没有self的方法称为类方法,而把带有self的方法称为实例方法,根据上面的描述可以发现,这种划分是错误的。
那么,Python中有没有类方法呢?
答案是,有。那么如何定义一个类方法呢?

Pyhon 类方法

还是使用上面的例子

>>> class Foo(object):
...     @classmethod                #定义类方法要点1
...     def foo(cls):               #定义类方法要点2
...             print ‘call foo‘
...
>>> Foo.foo()
call foo
>>> Foo().foo()
call foo

定义类方法需要注意两点:1. 添加@classmethod;2. 添加cls参数这样定义的类方法可以通过类名.方法名()的形式调用,也可以通过实例.方法名()的形式调用。

看到这里会发现,在Python中定义方法,总要带两个参数self或者cls。其中通过self限定的method必须使用实例才能调用。
那么很自然的一个疑问是,能不能定义不包含self及cls的方法呢?像最开始的例子中foo()那样。答案是有的,办法就是加@staticmethod修饰器。
这种被@staticmethod修饰器修饰的方法,称为静态方法

Python 静态方法

除了类方法,还有静态方法,请看下面这个例子:

>>> class Foo(object):
...     @staticmethod
...     def foo():
...             print ‘call foo‘
...
>>> Foo.foo()
call foo
>>> Foo().foo()
call foo

静态方法可以通过类名.方法名()和实例.方法名()的形式调用。查看type结果如下:

>>> type(Foo.foo)
<type ‘function‘>

可以看到,静态方法的类型是function,而类方法的类型是instancemethod。

总结

最后来总结一下:
从Python方法定义的角度出发,可以分为三种:
1.第一个参数是self;
2.第一个参数是cls;
3.参数既不含self也不含cls的
对于第一种方法,必须通过实例.方法名()或类名.方法名(实例)的形式调用;
对于第二种,可以通过实例.方法名()或类名.方法名()的形式调用,不能通过类名.方法名(实例)的形式调用;
对于第三种,方法即是普通函数,但是必须通过实例.方法名()或类名.方法名()的形式调用,不能通过其他形式调用

时间: 2024-08-10 02:10:20

[转载]Python方法绑定——Unbound/Bound method object的一些梳理的相关文章

使用MethodType函数将方法绑定到类或实例上

在开始正文之前,需要了解下Python的绑定方法(bound method)和非绑定方法. 简单做个测试: 定义一个类,类中由实例方法.静态方法和类方法. class ClassA: def instance_method(self): print('instance_method', self) @classmethod def cls_method(cls): print('cls_method', cls) @staticmethod def static_method(): print(

python tips:类的绑定方法(bound)和非绑定方法(unbound)

类属性只有类及其实例能够访问,可以理解为一个独立的命名空间. Python中类属性的引用方式有两种: 1. 通过类的实例进行属性引用,称为绑定方法(bound method),可以理解为方法与实例绑定在一起. 2. 通过类进行属性引用,称为非绑定方法(unbound method),方法没有与实例绑定. 在绑定方法中,为了与实例绑定,Python自动将实例作为方法的第一个参数,而非绑定方法则表现的像普通函数,和普通函数的区别在于它只能通过类来访问. 两种引用方式: 1 class A: 2 de

Python 中的方法、静态方法(static method)和类方法(class method)

英文原文: https://julien.danjou.info/blog/2013/guide-python-static-class-abstract-methods翻译出处: http://python.jobbole.com/81595/ 一.How methods work in Python 方法就是一个函数.以类的属性被存储.可以通过如下的形式进行声明和访问: In [1]: class Pizza(object):    ...:     def __init__(self,si

Python学习————绑定方法

一:Python多继承的正确打开方式:minins机制 Mixins核心:在多继承背景下,尽可能地提升多继承的可读性 让多继承满足人的思维习惯 ==> 什么 是 什么 class Vehicle: # 交通工具 def fly(self): print("I am flying") class CivilAircraft(Vehicle): # 民航飞机 pass class Helicopter(Vehicle): # 直升飞机 pass class Car(Vehicle):

cocos2dx 关于lua 绑定的环境配置官方文档翻译与 将自定义的方法绑定到lua的的方法

//网上有好多写怎样讲自定义的方法绑定到lua的文章,其中都只对环境配置做了简单的介绍,看到有的帖子写在绑定中遇到了各种各样的error,大部分是由于环境配置//不正确导致的,以下是官方的文档有标准的说明,所有的开发引擎都会有自己的说明文档.下面就是cocos2dx 官方文档 //怎样使用 bindings-generator How to Use bindings-generator ================== //windows 环境下 On Windows: ----------

TypeError: ‘method‘ object is not subscriptable

Python Django编程时遇到如下错误 File "C:\Users\shisheng\PycharmProjects\cdn2018\apps\cdnproject\views.py", line 23, in post urls = request.POST.get['aliurl'] TypeError: 'method' object is not subscriptable 注意下,这里应该使用()因为这是一个类的方法 request.POST.get('aliurl'

python问题:TypeError: a bytes-like object is required, not &#39;str&#39;

源程序: import socket target_host = "www.baidu.com" # 127.0.0.1 target_port = 80 # 建立一个socket对象 client = socket.socket(socket.AF_INET,socket.SOCK_STREAM) # 创建TCP连接 # 连接客户端 client.connect((target_host,target_port)) client.send("GET / HTTP/1.1\r

重构改善既有代码设计--重构手法08:Replace Method with Method Object (以函数对象取代函数)

你有一个大型函数,其中对局部变量的使用,使你无法釆用 Extract Method. 将这个函数放进一个单独对象中,如此一来局部变量就成了对象内的值域(field) 然后你可以在同一个对象中将这个大型函数分解为数个小型函数. class Order... double price() { double primaryBasePrice; double secondaryBasePrice; double tertiaryBasePrice; // long computation; ... }

python 3.5: TypeError: a bytes-like object is required, not &#39;str&#39;

出现该错误往往是通过open()函数打开文本文件时,使用了'rb'属性,如:fileHandle=open(filename,'rb'),则此时是通过二进制方式打开文件的,所以在后面处理时如果使用了str()函数,就会出现该错误,该错误不会再python2中出现. 具体解决方法有以下两种: 第一种,在open()函数中使用'r'属性,即文本方式读取,而不是'rb',以二进制文件方式读取,可以直接解决问题. 第二种,在open()函数中使用'rb',可以在使用之前进行转换,有以下实例,来自:htt