python动态类函数和静态类函数的区别

文章比较了动态类装饰器和静态类装饰器在Python中的行为差异,静态类装饰器在导入时读取,而动态类装饰器在实例化时读取。通过示例展示了如何在子类中修改静态和动态类的列表输出结果。

动态类装饰器函数在调用时读取,而静态类装饰器函数在导入时读取。这种差别引申出了两种类装饰器的不用适用场景。下面通过示例代码来直观的演示两种装饰器的区别。

静态类

class Test:
    list = ['0']
    @staticmethod
    def out():
        print(Test.list)

class child1(Test):
    list = ['1', '2']

class child2(Test):
    list = ['1', '5']
    
a = child1()
a.out()
b = child2()
b.out()

上述代码执行后,a.out()和b.out()输出结果都是[‘0’]。原因就是静态类函数out()调用的是父类Test的list,所以通过子类给list赋值的做法是行不通的,若想在子类改变out()的输出就必须在子类里重定义out()函数(见第三个示例)。通过在子类里给父类的list重新赋值的做法一样行不通,参见第四个示例。

动态类

class Test:
    list = []
    @classmethod
    def out(cls):
        print(cls.list)

class child1(Test):
    list = ['1', '2']

class child2(Test):
    list = ['1', '5']
    
a = child1()
a.out()
b = child2()
b.out()

上述代码执行后,a.out()输出结果为[‘1’, ‘2’],b.out()输出结果为[‘1’, ‘5’]。因为动态类函数指向的是子类本身的list,所以在子类里给list赋值后,out()函数输出的就是子类自己的list。

第三个示例

class Test:
    list = ['0']
    @staticmethod
    def out():
        print(Test.list)

class child1(Test):
    list = ['1', '2']
    @staticmethod
    def out():
        print(child1.list)

class child2(Test):
    list = ['1', '5']
    @staticmethod
    def out():
        print(child2.list)

child1.out()
child2.out()

该示例输出结果是[‘1’, ‘2’]和[‘1’, ‘5’]。

第四个示例

class Test:
    list = []
    @staticmethod
    def out():
        print(Test.list)
        
class child1(Test):
    Test.list = ['1', '5']

class child2(Test):
    Test.list = ['1', '2']

child1.out()

该示例输出结果为[‘1’, ‘2’],什么原因?为什么不是[‘1’, ‘5’]?因为子类本身没有out()函数,向上查找到父类Test有该函数,但该函数为静态函数,在文件载入时就已经读取至内存,读取过程中搜索Test.list变量的值,于是处于最后面的child2里的变量值便被采用。

### Python 中实例方法、方法静态方法的区别Python 中,中的方法可以根据其绑定的对象型分为三:实例方法(Instance Methods)、方法(Class Methods)静态方法(Static Methods)。这三种方法各自有着不同的特性用途[^1]。 #### 实例方法(Instance Methods) 实例方法是默认的方法型。它必须有一个名为 `self` 的参数(尽管可以使用其他名称),该参数指向调用该方法的实例对象。通过实例方法,可以访问修改实例的状态。 示例代码: ```python class MyClass: def __init__(self, value): self.value = value def instance_method(self, a): return self.value + a # 访问实例变量 obj = MyClass(5) print(obj.instance_method(3)) # 输出 8 ``` 实例方法需要通过实例来调用,无法直接通过名调用。如果尝试通过名调用实例方法,Python 会抛出错误,因为缺少 `self` 参数[^3]。 --- #### 方法(Class Methods) 方法通过装饰器 `@classmethod` 定义。它必须有一个名为 `cls` 的参数(尽管可以使用其他名称),该参数指向本身而非实例。通过方法,可以访问修改的状态,而不需要创建实例。 示例代码: ```python class MyClass: class_variable = 0 @classmethod def class_method(cls, a): cls.class_variable += a # 修改变量 return cls.class_variable print(MyClass.class_method(5)) # 输出 5 print(MyClass.class_method(3)) # 输出 8 ``` 方法可以通过名或实例调用,但无论哪种方式,`cls` 参数始终指向本身。因此,方法非常适合用于工厂模式或需要操作级别的属性时[^2]。 --- #### 静态方法(Static Methods) 静态方法通过装饰器 `@staticmethod` 定义。它既不接收 `self` 参数也不接收 `cls` 参数,因此它与或实例无关。静态方法更像是一个普通的函数,只是它被组织在的命名空间中。 示例代码: ```python class MyClass: @staticmethod def static_method(a, b): return a + b # 不依赖或实例的状态 print(MyClass.static_method(3, 4)) # 输出 7 obj = MyClass() print(obj.static_method(3, 4)) # 输出 7 ``` 静态方法可以通过名或实例调用,但它不会自动传递任何隐式的参数(如 `self` 或 `cls`)。因此,静态方法通常用于将逻辑分组到中,而不涉及或实例的状态[^2]。 --- ### 总结 - **实例方法**:绑定到实例,第一个参数为 `self`,用于访问修改实例状态。 - **方法**:绑定到,第一个参数为 `cls`,用于访问修改状态。 - **静态方法**:不绑定到或实例,无隐式参数,用于将逻辑组织到中。 通过合理选择这三种方法,可以提高代码的可读性灵活性[^1]。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值