一、动态语⾔的定义
动态语言是在运行时确定数据类型的语言。变量使用之前不需要类型声明,通常变量的类型是被赋值的那个值的类型。现在比较热门的动态语言有:python、PHP、javascript、Objective-C等,而 C 、 c++ 等语言则不属于动态语言。
二、运行的过程中给对象绑定(添加)属性
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
p = Person("小明","24")
print(p.name)
print(p.age)
运行结果为:
小明
24
这里我们只定义了name和age两个属性,但是在类已经定义好了之后,我们仍然可以往里面添加属性,这就是动态语言的好处,动态的给实例绑定属性:
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
p = Person("小明","24")
print(p.name)
print(p.age)
#动态添加属性
p.sex = "男"
print(p.sex)
运行结果为:
小明
24
男
三、运行的过程中给类绑定(添加)属性
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
P1 = Person("小明",24)
print(P1.sex)
运行结果为:
Traceback (most recent call last):
File "C:\Users\Se7eN_HOU\Desktop\test.py", line 8, in <module>
print(P1.sex)
AttributeError: 'Person' object has no attribute 'sex'
这是程序报错说,Person没有sex这个属性,我们可以通过给Person动态绑定属性,解决问题
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
P1 = Person("小明",24)
#动态给类添加属性
Person.sex = "男"
print(P1.sex)
这个时候在运行就不会出错,而且会打印出P1.sex为男
四、运行的过程中给类绑定(添加)方法
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
def eat(self):
print("正在吃东西")
P1 = Person("小明",24)
P1.eat()
P1.run()
运行结果为:
正在吃东西
Traceback (most recent call last):
File "C:\Users\Se7eN_HOU\Desktop\test.py", line 11, in <module>
P1.run()
AttributeError: 'Person' object has no attribute 'run'
说明:正在吃东西打印出来了,说明eat函数被执行,但是后面报错说没有run这个属性,但是我想在类创建好了以后,在运行的时候动态的添加run方法怎么办呢?
#动态添加方法需要导入types模块
import types
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
def eat(self):
print("正在吃东西")
#定义好需要动态添加的方法
def run(self):
print("在跑步")
P1 = Person("小明",24)
#正常调用类里面的函数
P1.eat()
#给对象动态绑定方法
P1.run = types.MethodType(run,P1)
#对象调用动态绑定的方法
P1.run()
运行结果为:
正在吃东西
在跑步
打印出来“在跑步”说明run方法被正常执行了
动态绑定类方法和静态方法
#动态添加方法需要导入types模块
import types
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
def eat(self):
print("正在吃东西")
#定义好需要动态添加的实例方法
def run(self):
print("在跑步")
#定义好需要动态添加的类方法
@claSSMethod
def dynamicClassMethod(cls):
print("这是一个动态添加的类方法")
#定义好需要动态添加的静态方法
@staticmethod
def dynamicStaticMethod():
print("这是一个动态添加的静态方法")
P1 = Person("小明",24)
#正常调用类里面的函数
P1.eat()
#给对象动态绑定方法
#MethodType(参数1,参数2)
#参数1:是动态绑定哪个方法,只写方法名即可
#参数2:是把这个方法动态的绑定给谁
P1.run = types.MethodType(run,P1)
P1.run()
#动态绑定类方法的使用
Person.dynamicClassMethod = dynamicClassMethod
Person.dynamicClassMethod()
#动态绑定静态方法的使用
Person.dynamicStaticMethod = dynamicStaticMethod
Person.dynamicStaticMethod()
总结:
- 给对象绑定属性直接在使用前进行赋值使用即可
- 给对象动态绑定方法需要import types模块
- 给对象动态绑定实例方法,需要使用type.MethodType()方法
- 给类添加类方法和静态方法,也是直接在使用前赋值即可使用
五、运行的过程中删除属性、方法
删除的方法:
- del 对象.属性名
- delattr(对象, "属性名")
class Person(object):
def __init__(self,name=None,age=None):
self.name=name
self.age=age
P1 = Person("小明",24)
print("---------删除前---------")
print(P1.name)
del P1.name
print("---------删除后---------")
print(P1.name)
运行结果为:
---------删除前---------
小明
---------删除后---------
print(P1.name)AttributeError: 'Person' object has no attribute 'name'
六、__slots__
动态语言:可以在运行的过程中,修改代码
静态语言:编译时已经确定好代码,运行过程中不能修改
如果我们想要限制实例的属性怎么办?比如,只允许对Person实例添加name和age属性。
为了达到限制的目的,Python允许在定义class的时候,定义一个特殊的__slots__变量,来限制该class实例能添加的属性:
class Person(object):
__slots__=("name","age")
p = Person()
p.name = "老王"
p.age = 40
print(p.name)
print(p.age)
#slots之外的属性
p.sex = "男"
print(p.sex)
运行结果为:
老王
40
p.sex = "男"
AttributeError: 'Person' object has no attribute 'sex'
注意:
- 使用__slots__要注意,__slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的