zoukankan      html  css  js  c++  java
  • python @classmethon @staticmethod 的使用及区别

    python @classmethon @staticmethod 的使用及区别

    lewis_bo

    展开

    转自:http://www.zhihu.com/question/20021164

    类中最常用的方法是实例方法, 即通过通过实例作为第一个参数的方法。
    举个例子,一个基本的实例方法就向下面这个:

    class Kls(object):
    
    def __init__(self, data):
        self.data = data
    
    def printd(self):
        print(self.data)
    
    ik1 = Kls('arun')
    ik2 = Kls('seema')
    ik1.printd()
    ik2.printd()


    这会给出如下的输出:
    seema



    然后看一下代码和示例图片:

    • 1,2参数传递给方法.
    • 3 self参数指向当前实例自身.
    • 4 我们不需要传递实例自身给方法,Python解释器自己会做这些操作的.

    如果现在我们想写一些仅仅与类交互而不是和实例交互的方法会怎么样呢? 我们可以在类外面写一个简单的方法来做这些,但是这样做就扩散了类代码的关系到类定义的外面. 如果像下面这样写就会导致以后代码维护的困难:

    1.  
    2. def get_no_of_instances(cls_obj):

    3. return cls_obj.no_inst

    4. class Kls(object):

    5. no_inst = 0

    6. def __init__(self):

    7. Kls.no_inst = Kls.no_inst + 1

    8. ik1 = Kls()

    9. ik2 = Kls()

    10. print(get_no_of_instances(Kls))


    输出:
    2
    @classmethod
    我们要写一个只在类中运行而不在实例中运行的方法. 如果我们想让方法不在实例中运行,可以这么做:

    
     
    1.  
    2. def iget_no_of_instance(ins_obj):

    3. return ins_obj.__class__.no_inst

    4. class Kls(object):

    5. no_inst = 0

    6. def __init__(self):

    7. Kls.no_inst = Kls.no_inst + 1

    8. ik1 = Kls()

    9. ik2 = Kls()

    10. print iget_no_of_instance(ik1)


    输出
    2
    在Python2.2以后可以使用@classmethod装饰器来创建类方法.

    
     
    1.  
    2. class Kls(object):

    3. no_inst = 0

    4. def __init__(self):

    5. Kls.no_inst = Kls.no_inst + 1

    6. @classmethod

    7. def get_no_of_instance(cls_obj):

    8. return cls_obj.no_inst

    9. ik1 = Kls()

    10. ik2 = Kls()

    11. print ik1.get_no_of_instance()

    12. print Kls.get_no_of_instance()


    输出:
    2
    2
    这样的好处是: 不管这个方式是从实例调用还是从类调用,它都用第一个参数把类传递过来.
    @staticmethod
    经常有一些跟类有关系的功能但在运行时又不需要实例和类参与的情况下需要用到静态方法. 比如更改环境变量或者修改其他类的属性等能用到静态方法. 这种情况可以直接用函数解决, 但这样同样会扩散类内部的代码,造成维护困难.
    比如这样:

    
     
    1.  
    2. IND = 'ON'

    3. def checkind():

    4. return (IND == 'ON')

    5. class Kls(object):

    6. def __init__(self,data):

    7. self.data = data

    8. def do_reset(self):

    9. if checkind():

    10. print('Reset done for:', self.data)

    11. def set_db(self):

    12. if checkind():

    13. self.db = 'new db connection'

    14. print('DB connection made for:',self.data)

    15. ik1 = Kls(12)

    16. ik1.do_reset()

    17. ik1.set_db()


    输出:
    Reset done for: 12
    DB connection made for: 12
    如果使用@staticmethod就能把相关的代码放到对应的位置了.

    
     
    1.  
    2. IND = 'ON'

    3. class Kls(object):

    4. def __init__(self, data):

    5. self.data = data

    6. @staticmethod

    7. def checkind():

    8. return (IND == 'ON')

    9. def do_reset(self):

    10. if self.checkind():

    11. print('Reset done for:', self.data)

    12. def set_db(self):

    13. if self.checkind():

    14. self.db = 'New db connection'

    15. print('DB connection made for: ', self.data)

    16. ik1 = Kls(12)

    17. ik1.do_reset()

    18. ik1.set_db()


    输出:
    Reset done for: 12
    DB connection made for: 12
    下面这个更全面的代码和图示来展示这两种方法的不同
    @staticmethod 和 @classmethod的不同

    
     
    1.  
    2. class Kls(object):

    3. def __init__(self, data):

    4. self.data = data

    5. def printd(self):

    6. print(self.data)

    7. @staticmethod

    8. def smethod(*arg):

    9. print('Static:', arg)

    10. @classmethod

    11. def cmethod(*arg):

    12. print('Class:', arg)

    13.  
    14. >>> ik = Kls(23)

    15. >>> ik.printd()

    16. 23

    17. >>> ik.smethod()

    18. Static: ()

    19. >>> ik.cmethod()

    20. Class: (<class '__main__.Kls'>,)

    21. >>> Kls.printd()

    22. TypeError: unbound method printd() must be called with Kls instance as first argument (got nothing instead)

    23. >>> Kls.smethod()

    24. Static: ()

    25. >>> Kls.cmethod()

    26. Class: (<class '__main__.Kls'>,)

    27.  


    下面这个图解释了以上代码是怎么运行的:

  • 相关阅读:
    从Java到C++——常量的使用规则
    LintCode 二叉树的遍历 (非递归)
    POJ 3592 Instantaneous Transference(强连通+DP)
    怎样给UINavigationBar加入button?
    《Spring技术内幕》笔记-第四章 Spring MVC与web环境
    HDU 4714 Tree2cycle(树型DP)
    hdu 1102 Constructing Roads(kruskal || prim)
    [Android随笔]内存泄漏以及内存溢出
    保存数据同一时候查询保存数据记录的ID
    8086的储存器编址
  • 原文地址:https://www.cnblogs.com/grj001/p/12222956.html
Copyright © 2011-2022 走看看