staticmethod和classmethod
类的实例化基本遵循创建实例对象、初始化实例对象、最后返回(产生)实例对象这么一个过程
1 | class Test(object): |
共同:
- 两个都是装饰器,装饰的成员函数可以通过
类名.方法名(...)
来调用
区别:
- ▲最显著的特点是
classmethod
需要传递一个参数cls
,而staticmethod
不需要。因此可以访问、修改类的属性,类的方法,实例化对象等,避免硬编码;而staticmethod
不行 classmethod
可以判断出自己是通过基类被调用,还是通过某个子类被调用
用途:
classmethod
多用于设计模式之工厂模式,将解析逻辑封装在方法本身内部。(也可认为构造前交互,即在进行实例化类对象之前先进行某些逻辑操作,即可看作进行不同的构造函数,然后返回一个类实例,见具体代码),更多拓展见附录__new__
和__init__
staticmethod
用法就跟其他的语言中的静态static用法相同(可看作是属于该类的一个工具、辅助函数)
具体代码
1 | # Python program to demonstrate |
可看下这篇class method vs static method in Python
附录__new__
和__init__
知识点
▲1. 首先要弄懂调用顺序
__new__
–>__init__
: 所以__init__
其实不是实例化一个类的时候第一个被调用 的方法。当使用 Persion(name, age) 这样的表达式来实例化一个类时,最先被调用的方法 其实是__new__
方法。
__new__
至少要有一个参数cls,代表要实例化的类,此参数在实例化时由Python解释器自动提供(实例化时加的参数,也会被认为时cls的属性,见①)。即系统知道__new__
() 方法始终都是类的类方法,即使没有被加上类方法装饰器。
__new__
方法接受的参数虽然也是和__init__
一样,但__init__
是在类实例创建之后调用,而__new__
方法正是创建这个类实例的方法。▲2.
__init__
有一个参数self,就是这个__new__
中return的实例,__init__
在__new__
的基础上可以完成一些其它初始化的动作,__init__
不需要返回值。而__new__
必须有(可以return父类__new__
出来的实例,或者直接是object的__new__
出来的实例)。__new__
方法决定了创建哪个类的实例(可以是父类,也可以是子类),因此不一定调用当前类的__init__
。即(若__new__
没有正确返回当前类cls的实例,那__init__
是不会被调用的,即使是父类的实例也不行)3.继承自object的新式类才有
__new__
,同时,在任何新式类的__new__
()方法,不能调用自身的__new__
()来制造实例,因为这会造成死循环,即class Foo(object)
的__new__
中不能Foo.__new__(cls, *args, **kwargs)。
区别
1.__init__
通常用于初始化一个新实例,控制这个初始化的过程,比如添加一些属性, 做一些额外的操作,发生在类实例被创建完以后。它是实例级别的方法。
2.__new__
通常用于控制生成一个新实例的过程。它是类级别的方法。
依照Python官方文档的说法,__new__
方法主要是当你继承一些不可变的class时(比如int, str, tuple), 提供给你一个自定义这些类的实例化过程的途径。还有就是实现自定义的metaclass。
1 | class A(object): # 新式类:继承自object |
总结: a = A() --> __new__
方法,return调用父类__new__
生成一个类实例对象 -->__init__(self,...)
中的self实例就是这个父类__new__
出来的实例
__new__
() 是在新式类中新出现的方法,它作用在构造方法建造实例之前,可以这么理解,在 Python 中存在于类里面的构造方法__init__
() 负责将类的实例化,而在__init__
() 启动之前,__new__
() 决定是否要使用该__init__
() 方法,因为__new__
() 可以调用其他类的构造方法或者直接返回别的对象来作为本类的实例。
用__new__
来实现设计模式中的–单例模式
只能实例化一个类对象
1 | class Singleton(object): |
▲ Python所有类都继承自<class 'type'>
,包括新式类object
Author: Mrli
Link: https://nymrli.top/2019/07/07/staticmethod和classmethod区别/
Copyright: All articles in this blog are licensed under CC BY-NC-SA 3.0 unless stating additionally.