- 主题:functools.partial 和 partialmethod 的困惑
python中的函数(也不仅仅是函数)和c中的不太一样,函数本身其实不是callable。函数只是描述了自己是干什么的、接受什么参数等等,既然是描述顾名思义是个descriptor。一个函数真的被调用时python内部隐含了一个根据函数的描述生成一个符合描述的对象的过程,生成的这个对象才是callable的。
class A(object):
def foo(self, arg):
这里foo只是一个描述,不是一个实际的对象(同时,A也是)。
a = A()
根据A的描述,一个对象a生成了,生成的时候,A描述了内部有个foo,对象a内部也生成一个foo。
print(A.foo)
print(a.foo)
会看到一个是funtion,一个是bound method。
至于partial和partialmethod,如官方文档所言,partial生成了一个callable,而partialmethod并没有,只是生成了一个如何生成callable的descriptor。
原始代码可以,是因为本来getMe就可以是一个descriptor,此时其地位和callAPI是一样的。后续使用时在生成Client的对象时其描述中也带了getMe,会生成一个getMe bound method。
方案1不可以,是因为在getattr这个范围和在全局范围中均找不到callAPI这个符号。注意传统c中的成员函数,在python中不在符号表里。顺便说原始的能找到符号是因为其在Client这个范围内。
方案2不可以的原因在前面说了,desc和func的区别以及和实例化对象的关系。
【 在 JulyClyde 的大作中提到: 】
: 我在写一个 telegram bot ,自己寨了一个小的 lib
: ```
: class Client:
: ...................
--
FROM 114.246.236.*
(下面都是我个人的理解,不严谨啊)
按照传统c++的思路:
class A {
void foo(int) ...
得到的程序里foo是个函数,其代码放在内存1234位置,调用该函数时就jmp 1234;内部使用this时该指针指向一个对象,此时对象与1234处的代码块无关(重点是这个“无关”)。
但是python里不是这样子的:
class A
def foo(self ...
人眼看到的貌似可以执行的函数foo在解释器眼里是一段desc,解释了当需要执行此段脚本时需要干什么。
当生成一个对象a时,解释器根据对foo的描述,生成一个“真的函数” bound method,作为a的一个方法,这个方法是callable的。(不过这个只是“逻辑生成”,真正跑的代码不会每个__dict__都包含所有的属性方法)
换个说法的话,def foo这个def本身可以视为没有__call__能力,不callable。通过这个def,解释器自动生成了一个obj,其中obj.__call__执行了def定义的脚本内容,这个obj“被叫成foo给人看”,然后这个obj(在逻辑上)绑定到对象上,在对象点操作时(在逻辑上)从__dict__中拿到它(即obj,而不是def)
比如:
class A(object):
def foo(self):
print("original")
def foo(self):
print("tmp")
a = A()
b = A()
b.foo = types.MethodType(foo, b)
a.foo()
b.foo()
print(a.foo)
print(b.foo)
print(a.__dict__)
print(b.__dict__)
至于partial和partialmethod,则是一个生成callable,一个生成desc。当解释器看到desc时,就根据描述符协议生成“人类函数”,类似前面说的def生成obj。
比如:
class A(object):
def foo(self, a):
return a
foo2 = partialmethod(foo, 2)
a = A()
print(A.foo2)
print(a.foo2)
【 在 JulyClyde 的大作中提到: 】
: 函数本身并不是callable??啊?这个咋理解?
A: 问题是:它其实是callable啊
: 象
: ...................
--
FROM 114.246.237.*
我觉得是这个意思(不过其实我也不熟悉这块,理解可能不对)
有一个小点是method是通过desc生成的,生成后的这个method是个callable。
【 在 JulyClyde 的大作中提到: 】
: 嗯,我尝试理解一下你的意思:
: 虽然
: class下边的def出来的东西是个callable,同时也是descriptor
: ...................
--
FROM 124.64.22.*