python实现静态变量

在类中定义在函数外面的变量是类变量,不属于类的实例。利用它可以实现静态变量。

1. 意料之外的静态变量

在使用类变量的时候一定要小心,否则会得到意料之外的结果。看下面的代码:

class Foo(object):
    count = 0

f1 = Foo()
f2 = Foo()
f1.count = 1
print(f1.count, f2.count)
# 结果:
1 0

之所以结果不相同,原因在于第6行赋值的时候,并没有把类变量count变为1,而是f1对象自己生成了一个变量count,并初始化为1。而此时f2.count指向的仍然是类Foo的变量count,其值仍为0。

2. python动态语言特性

我们来看以下代码就明白了:

class test(object):
    pass

t = test()
print(t.count) # 会出错,因为count不存在
t.count = 1
print(t.count) # 不会出错,因为新创建了一个count

3. 通过类名来引用类变量

所以要把最开始的代码改成这样才不会出问题:

class Foo(object):
    count = 0

f1 = Foo()
f2 = Foo()
Foo.count = 1
print(f1.count, f2.count)
# 结果:
1 1

可以用getter和setter来进行包装:

class Foo(object):
    count = 0

    def get_count(self):
        return Foo.count

    def set_count(self, num):
        Foo.count = num

f1 = Foo()
f2 = Foo()
Foo.set_count(1)
print(f1.get_count(), f2.get_count())
# 结果:
1 1

为了支持在类的实例中操作静态变量,我们可以借助@property装饰器来这样写:

class Foo(object):
    _count = 0 # 不要直接操作这个变量,也尽量避免访问它

    @property
    def count(self):
        return Foo._count

    @count.setter
    def count(self, num):
        Foo._count = num


f1 = Foo()
f2 = Foo()
print f1.count, f1._count, f2.count, f2._count

f1.count = 1
print f1.count, f1._count, f2.count, f2._count
# 结果:
0 0 0 0
1 1 1 1

4. 类变量的封装

为了避免直接在外部对类变量进行操作,我们可以再进行封装,把类变量加双下划线__变成私有的,再使用@classmethod来实现:

class Foo(object):
    __count = 0 # 私有变量,无法在外部访问,Foo.__count会出错

    @classmethod
    def get_count(cls):
        return cls.__count

    @classmethod
    def set_count(cls, num):
        cls.__count = num

f1 = Foo()
f2 = Foo()
Foo.set_count(1)
print(f1.get_count(), f2.get_count())
# 结果:
1 1

值得注意的是,此时__count是属于类Foo的私有变量,在类的外部无法访问私有变量__count,即使在类的内部也无法通过普通函数访问,甚至无法通过Foo.__count访问, 只能通过@classmethod来访问

5. 扩展问题:如何将classmethod变成property?

直接在@classmethod前面加@property会报错:

TypeError: ‘classmethod’ object is not callable

实现起来比较复杂,目前来看有两种思路。

一种是实现一个property的子类,然后修改其 __get__ 方法。
另一种方式是使用元类 (__metaclass__)。

关于这一问题的讨论参见stackoverflow:

### Python 类中静态变量的定义 在Python中,类属性可以被视作静态变量[^2]。这些属性属于类本身而不是特定的对象实例。因此,在整个程序运行期间,所有对象共享同一份数据。 #### 定义方式 要定义一个静态变量(即类级别的变量),只需在类内部但在任何方法之外声明它即可: ```python class MyClass: class_variable = 0 # 这是一个静态/类变量 def __init__(self, instance_value): self.instance_variable = instance_value # 实例变量 ``` 这里`MyClass.class_variable`就是所谓的“静态变量”,而`instance_variable`则是普通的实例成员。 ### 使用场景 当希望某些数据在整个应用程序生命周期内保持一致,并且不随各个对象的状态变化时,就可以考虑使用此类变量。例如计数器、配置参数等都适合用这种方式存储。 另一个常见的例子是在单件模式(Singleton Pattern)下保存唯一实例引用: ```python class SingletonExample: _singleton_instance = None # 静态变量用于保存唯一的实例 @staticmethod def get_instance(): if not SingletonExample._singleton_instance: SingletonExample._singleton_instance = SingletonExample() return SingletonExample._singleton_instance ``` 此代码片段展示了如何利用静态变量 `_singleton_instance` 来确保 `SingletonExample` 只有一个实例存在。 ### 特性 - **共享性质**: 所有由同一个类创建出来的对象都会访问相同的静态变量副本; - **初始化时机**: 当加载含有该类定义模块的时候就会自动完成初始化工作; - **内存效率高**: 不管有多少个对象实例化出来,静态变量只占用一份空间; - **修改影响广泛**: 对于可变类型的静态变量来说,一旦其值发生改变将会反映到每一个依赖它的地方上去。 需要注意的是,尽管上述特性使得静态变量非常有用,但也可能带来意想不到的行为特别是涉及到并发编程的情况下。所以在实际开发过程中应当谨慎对待这类设计决策。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值