【python】classmethod & staticmethod 区别

来源:http://blog.csdn.net/carolzhang8406/article/details/6856817

其他参考:

http://blog.csdn.net/lovingprince/article/details/6595466

http://blog.csdn.net/handsomekang/article/details/9615239

http://python.jobbole.com/83584/

比较好的讨论:

http://bbs.csdn.net/topics/350141376

https://www.zhihu.com/question/31844003

 

classmethod:类方法
staticmethod:静态方法

在python中,静态方法和类方法都是可以通过类对象和类对象实例访问。但是区别是:

  • @classmethod 是一个函数修饰符,它表示接下来的是一个类方法,而对于平常我们见到的则叫做实例方法。 类方法的第一个参数cls,而实例方法的第一个参数是self,表示该类的一个实例。 
  • 普通对象方法至少需要一个self参数,代表类对象实例
  • 类方法有类变量cls传入,从而可以用cls做一些相关的处理。并且有子类继承时,调用该类方法时,传入的类变量cls是子类,而非父类。 对于类方法,可以通过类来调用,就像C.f(),有点类似C++中的静态方法, 也可以通过类的一个实例来调用,就像C().f(),这里C(),写成这样之后它就是类的一个实例了。 
  • 静态方法则没有,它基本上跟一个全局函数相同,一般来说用的很少

Example 1:

>>> class a():

@staticmethod
def staticm():
print 'static'
def normalm(self):
print 'nomarl',self
@classmethod
def classm(cls):
print 'class',cls


>>> a1=a()
>>> a1.normalm()
nomarl <__main__.a instance at 0x84dddec>
>>> a1.staticm()
static
>>> a1.classm()
class __main__.a
>>> type(a)
<type 'classobj'>
>>> type(a1)
<type 'instance'>



Example 2:

class A(object):
@classmethod
def cm(cls):
print '类方法cm(cls)调用者:', cls.__name__
@staticmethod
def sm():
print '静态方法sm()被调用'

class B(A):
pass

A.cm()
B.cm()

A.sm()
B.sm()

输出:

类方法cm(cls)调用者: A
类方法cm(cls)调用者: B
静态方法sm()被调用
静态方法sm()被调用

 

 

#coding=utf-8

class Test(object):
 
    def InstanceFun(self):
        print("InstanceFun");
        print(self);
 
    @classmethod
    def ClassFun(cls):
        print("ClassFun");
        print(cls);
 
    @staticmethod
    def StaticFun():
        print("StaticFun");
    
    def Ballshurt():  #该函数怎么调用都是错的
        print("Ballshurt");
 

t = Test();
t.InstanceFun();
print "---------"
Test.ClassFun();
print "---------"
Test.StaticFun();
print "---------"
t.StaticFun();
print "---------"
t.ClassFun();
print "---------"
Test.InstanceFun(t);
print "---------"
Test.InstanceFun(Test);
print "---------"

#Test.Ballshurt();  #error
#t.Ballshurt(); #access error

看上面例子,当类中的函数不添加参数时是错误的,没有办法通过实例或类本身调用,而@staticmethod修饰后就可以访问了。

 

 

 

一些网上观点:

实例方法隐含的参数为类实例,而类方法隐含的参数为类本身。
静态方法无隐含参数,主要为了类实例也可以直接调用静态方法。
所以逻辑上类方法应当只被类调用,实例方法实例调用,静态方法两者都能调用。
注意啊只是逻辑上啊,其实python的类模型实现上灵活而蛋疼着。
这些玩意基本都是为了隐含参数而搞出来的鬼名堂。
了解这点只要参数个数匹配,python就允许你"胡扯"了。

----------------------------------------

Python的类就是个语法糖。一个函数写在类里面和写在类外面没有区别,唯一的区别就是参数,所谓实例方法就是第一个参数是self,所谓类方法就是第一个参数是class,而静态方法不需要额外的参数,所以必须区分。

-----------------------------------------

如果你看过装饰器staticmethod的代码就会发现,它唯一的作用不过是加了个不用的self参数而已。python里并没有真正意义上的静态方法,因为类定义本身也是个实例(继承自object的为type实例,裸的为classobj实例),因此一个“静态方法”其实只不过是类定义对象的成员罢了……

c++的static方法则是正儿八经的静态方法,因为类定义本身不是个实例。

python里,除了极少的几个语句,其余全都是对象实例。一个函数同样是对象实例,因此对python而言其实连方法这一说都不准确…那些个对象方法只不过是指向函数对象的引用,只是属性而已……


posted @ 2016-10-25 20:47  匡子语  阅读(1352)  评论(0编辑  收藏  举报