最近在思考__call__和类方法的使用场景,虽然功能会有重叠,但凡存在的必是合理的,肯定会有轻微的不同,看了django以及部分框架的设计源码,各有各的设计风格,本着功能边界的看法,所以做个总结。
class A:
def __init__(self, a, b):
self.a = a
self.b = b
def do_staic_something(self, c ,d):
print("123")
@staticmethod
def do_something():
pass
@classmethod
def do_class_something(cls):
pass
def __call__(self):
print("123call")
实例化方法
实例化的方法就是普通的类方法,它表达了类功能的多样性和抽象性,传入参数是self,使用场景必须是先实例化,在调用此方法。但实例化方法也是调用call,就是a.do_something.__call()。
静态方法
staticmethod也是一个类方法,是可以直接类调用的。个人认为的使用场景是:只要要定义的方法里不涉及到self参数,就用静态方法承担。因为这样就表明这个方法和本身的类没有关系,明确的区别出类相关和不相关。
类方法
classmethod是区别与staticmethod的又一个类方法,也是类直接可调用的。它接受的第一个参数是cls,传入类的用途就是为了init不足而构造新对象。比如工厂模式,不断的用classmethod方法创造新产品,每个classmethod就像一条流水线,生产出来新的产品对象,供使用者拿去使用。而且因为不用实例化,写法也比较优雅。
另外,classmethod可以作为实现的窗口函数。先将要实现的功能全部抽象化,然后写在调用处实现窗口。之后通过子类的重写参数以及修改需要的逻辑方法,就可以在不改变代码的情况下通过继承来展示类的多态,也符合对象的多继承少修改原则。栗子比如django的View类中的as_view。
__call __
魔法方法call,相当于给类加上了callable属性,和普通函数一样可以被调用。个人认为的使用场景:
- 用类实现装饰器的时候。
- 功能单一,实现目的简单,轻量化。
同样可以用作抽象窗口函数,抽象出使用规则,通过子类的改写其他方法,在不改变原代码的情况下取得不同的结果。举个小栗子。
import pandas as pd
import os
from functools import wraps
class CSVDataProcess:
def __init__(self):
def example_one(self):
pass
def __call__(self, exmaple):
try:
target_func = getattr(self, exmaple)
target_func()
except Exception as e:
print("execute error", e)
if __name__ == "__main__":
csv = CSVDataProcess()
csv("example_one")