经常使用@classmethod
,但是对它的原理不甚明白,函数装饰器原理不是很复杂
1 那么classmethod
作为类装饰器,其作用流程
是怎样的?
附classmethod源码
class classmethod(object):
"""
classmethod(function) -> method
Convert a function to be a class method.
A class method receives the class as implicit first argument,
just like an instance method receives the instance.
To declare a class method, use this idiom:
class C:
def f(cls, arg1, arg2, ...): ...
f = classmethod(f)
It can be called either on the class (e.g. C.f()) or on an instance
(e.g. C().f()). The instance is ignored except for its class.
If a class method is called for a derived class, the derived class
object is passed as the implied first argument.
Class methods are different than C++ or Java static methods.
If you want those, see the staticmethod builtin.
"""
def __getattribute__(self, name): # real signature unknown; restored from __doc__
""" x.__getattribute__('name') <==> x.name """
pass
def __get__(self, obj, type=None): # real signature unknown; restored from __doc__
""" descr.__get__(obj[, type]) -> value """
pass
def __init__(self, function): # real signature unknown; restored from __doc__
pass
@staticmethod # known case of __new__
def __new__(S, *more): # real signature unknown; restored from __doc__
""" T.__new__(S, ...) -> a new object with type S, a subtype of T """
pass
__func__ = property(lambda self: object(), lambda self, v: None, lambda self: None) # default
这个链接解释的不错,你看下这里
http://stackoverflow.com/questions/16774...
Python文档里关于descriptor的说明有
descriptor
Any new-style object which defines the methods __get__(), __set__(), or __delete__().
When a class attribute is a descriptor, its special binding behavior is triggered upon attribute lookup. Normally,
using a.b to get, set or delete an attribute looks up the object named b in the class dictionary for a,
but if b is a descriptor, the respective descriptor method gets called. Understanding descriptors is a key to
a deep understanding of Python because they are the basis for many features including functions, methods,
properties, class methods, static methods, and reference to super classes.
For more information about descriptors’ methods, see descriptors.
所以对于classmethod封装的声明,等同于
class MyTest(object):
@classmethod
def sayHello(cls):
print "sayHello"
MyTest.sayHello()
cm = classmethod(MyTest)
cm.__get__(None, dict).im_func.sayHello()
你这段源码是哪里看到的?classmethod
应该是 built-in
用 C 写的吧,你贴的这段更像是 Python 模拟的版本。
真的源码:https://hg.python.org/cpython/file/69b41...