一、动态绑定属性和方法
在Python中,动态绑定属性和方法意味着你可以在运行时(即在对象的生命周期中)向一个对象添加或修改其属性和方法。
正常情况下,当我们定义了一个class
,创建了一个class
的实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性。
尝试给实例绑定一个属性:
尝试给实例绑定一个方法:
但是,给一个实例绑定的方法,对另一个实例是不起作用的:
为了给所有实例都绑定方法,可以给class
绑定方法,给class
绑定方法后,所有实例均可调用。
通常情况下,上面的set_score
方法可以直接定义在class
中,但动态绑定允许我们在程序运行的过程中动态给class
加上功能,这在静态语言中很难实现。
二、使用__slots__
如果我们想要限制实例的属性怎么办?比如,只允许对Student实例添加name
和age
属性。
Python允许在定义class
的时候,定义一个特殊的__slots__
变量,来限制该class
实例能添加的属性:
由于'sex'
没有被放到__slots__
中,所以不能绑定sex
属性,试图绑定sex
将得到AttributeError
的错误。
使用
__slots__
要注意,__slots__
定义的属性仅对当前类实例起作用,对继承的子类是不起作用的!!!除非在子类中也定义
__slots__
,这样,子类实例允许定义的属性就是自身的__slots__
加上父类的__slots__。
class Parent:__slots__ = ['parent_attr']def __init__(self, parent_value):self.parent_attr = parent_valueclass Child(Parent):__slots__ = ['child_attr']def __init__(self, parent_value, child_value):super().__init__(parent_value)self.child_attr = child_value# 创建实例 child_obj = Child(10, 20) print(child_obj.parent_attr) # 输出: 10 print(child_obj.child_attr) # 输出: 20# 尝试添加未在 __slots__ 中声明的属性 # child_obj.new_attr = 30 # 这将引发 AttributeError
三、__dict__
属性
__dict__
是一个字典属性,它存储了对象的所有属性和值。
对于类来说,__dict__
包含了类的静态函数、类函数、普通函数、全局变量以及一些内置的属性。而对于类的实例对象来说,__dict__
则包含了与实例相关的属性及其值。
class MyClass:class_var = 1def __init__(self, instance_var):self.instance_var = instance_varobj = MyClass(10)# {'__module__': '__main__', '__firstlineno__': 1, 'class_var': 1, '__init__': <function MyClass.__init__ at 0x0000014699813B00>, '__static_attributes__': ('instance_var',), '__dict__': <attribute '__dict__' of 'MyClass' objects>, '__weakref__': <attribute '__weakref__' of 'MyClass' objects>, '__doc__': None}
print(MyClass.__dict__) # 类的__dict__属性
# {'instance_var': 10}
print(obj.__dict__) # 实例对象的__dict__属性
【注意】:
对象的
__dict__
属性仅包含该对象自身的实例属性,而不包括从类继承的类属性。这是对象属性封装和继承机制的一部分。