由于这几个变量比较容易混淆,下面总结一下。
全局变量:
在一个py文件或者一个模块或者包中,但不在类中、不在函数内中的变量。
局部变量
在函数内、在class的方法内(未加self修饰),这就是局部变量
静态变量/类变量
在class内的,但不在class的方法内的,这就是静态变量/类变量(并且作为类被导入时,类变量是被初始化的)
实例变量
在class的方法内的,用self修饰的变量,这就是实例变量
看个例子:
global_a = 1
class MyClass:
class_a = 2 # 类变量
def __init__(self):
self.a = 3 # 实例变量
a = 4 # 局部变量
def sum(self):
global global_a # 函数内定义global_a变量,使用global来声明全局变量
global_a += 1
return global_a
c = MyClass()
1、全局变量与局部变量
global_a # 1
c.global_a # 实例c访问不到局部变量
Traceback (most recent call last):
File "<input>", line 1, in <module>
AttributeError: 'MyClass' object has no attribute 'global_a'
c.sum() # 2 由于类函数内使用global关键字来声明global_a,因此可以继承使用全局变量
global_a # 2 全局变量值被更新
2、类变量与实例变量(属性)
dir(c) # 内建函数dir()取出实例c拥有的类属性,和自己的实例属性
['__class__', '__delattr__', '__dict__', '__dir__', '__doc__', '__eq__', '__format__', '__ge__', '__getattribute__', '__gt__', '__hash__', '__init__', '__le__', '__lt__', '__module__', '__ne__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'a', 'class_a', 'sum']
c.__dict__ # __dict__定义个实例属性集,以key-value返回实例变量名称及值(vars(c)同样)
{'a': 3}
c.class_a # 2
c.a # 3
下面看下实例属性试图更改类属性的情况:
c.class_a = 666
c.class_a # 666 # 实例c只是更改了自己属性集中的class_a的value值,覆盖了类中继承来的class_a属性
MyClass.class_a # 2 # 类属性不会改变
但是如果类属性可变,情况就不同了:
class C:
x = {'a': 111}
c = C()
c.x # {'a': 111}
c.x.update({'b': 222})
c.x # {'a': 111, 'b': 222} 实例c的属性集更新
C.x # {'a': 111, 'b': 222} 类的属性字典x也被更新了
总结:
1、全局变量,类可以直接访问,函数内必须使用global关键字才可以访问全局变量
2、类属性和实例属性分别拥有自己的属性集,类属性更改会影响到所有实例,实例属性的更改,只是写入实例属性集,不影响类属性(但是可变类属性除外)。