上一篇文章為:→1.1.1元類
python是動(dòng)態(tài)語言
1. 動(dòng)態(tài)語言的定義
動(dòng)態(tài)編程語言
是 高級(jí)程序設(shè)計(jì)語言
的一個(gè)類別,在計(jì)算機(jī)科學(xué)領(lǐng)域已被廣泛應(yīng)用。它是一類 在運(yùn)行時(shí)可以改變其結(jié)構(gòu)
的語言 :例如新的函數(shù)、對(duì)象、甚至代碼可以被引進(jìn),已有的函數(shù)可以被刪除或是其他結(jié)構(gòu)上的變化。動(dòng)態(tài)語言目前非常具有活力。例如JavaScript便是一個(gè)動(dòng)態(tài)語言,除此之外如 PHP 、 Ruby 、 Python 等也都屬于動(dòng)態(tài)語言,而 C 、 C++ 等語言則不屬于動(dòng)態(tài)語言。----來自 維基百科
2. 運(yùn)行的過程中給對(duì)象綁定(添加)屬性
>>> class Person(object):
def __init__(self, name = None, age = None):
self.name = name
self.age = age
>>> P = Person("小明", "24")
>>>
在這里,我們定義了1個(gè)類Person,在這個(gè)類里,定義了兩個(gè)初始屬性name和age,但是人還有性別啊!如果這個(gè)類不是你寫的是不是你會(huì)嘗試訪問性別這個(gè)屬性呢?
>>> P.sex = "male"
>>> P.sex
'male'
>>>
這時(shí)候就發(fā)現(xiàn)問題了,我們定義的類里面沒有sex這個(gè)屬性啊!怎么回事呢? 這就是動(dòng)態(tài)語言的魅力和坑! 這里 實(shí)際上就是 動(dòng)態(tài)給實(shí)例綁定屬性
!
3. 運(yùn)行的過程中給類綁定(添加)屬性
>>> P1 = Person("小麗", "25")
>>> P1.sex
Traceback (most recent call last):
File "<pyshell#21>", line 1, in <module>
P1.sex
AttributeError: Person instance has no attribute 'sex'
>>>
我們嘗試打印P1.sex,發(fā)現(xiàn)報(bào)錯(cuò),P1沒有sex這個(gè)屬性!---- 給P這個(gè)實(shí)例綁定屬性對(duì)P1這個(gè)實(shí)例不起作用! 那我們要給所有的Person的實(shí)例加上 sex屬性怎么辦呢? 答案就是直接給Person綁定屬性!
>>>> Person.sex = None #給類Person添加一個(gè)屬性
>>> P1 = Person("小麗", "25")
>>> print(P1.sex) #如果P1這個(gè)實(shí)例對(duì)象中沒有sex屬性的話,那么就會(huì)訪問它的類屬性
None #可以看到?jīng)]有出現(xiàn)異常
>>>
4. 運(yùn)行的過程中給類綁定(添加)方法
我們直接給Person綁定sex這個(gè)屬性,重新實(shí)例化P1后,P1就有sex這個(gè)屬性了! 那么function呢?怎么綁定?
>>> class Person(object):
def __init__(self, name = None, age = None):
self.name = name
self.age = age
def eat(self):
print("eat food")
>>> def run(self, speed):
print("%s在移動(dòng), 速度是 %d km/h"%(self.name, speed))
>>> P = Person("老王", 24)
>>> P.eat()
eat food
>>>
>>> P.run()
Traceback (most recent call last):
File "<pyshell#5>", line 1, in <module>
P.run()
AttributeError: Person instance has no attribute 'run'
>>>
>>>
>>> import types
>>> P.run = types.MethodType(run, P)
>>> P.run(180)
老王在移動(dòng),速度是 180 km/h
既然給類添加方法,是使用類名.方法名 = xxxx,那么給對(duì)象添加一個(gè)方法也是類似的對(duì)象.方法名 = xxxx
完整的代碼如下:
import types
#定義了一個(gè)類
class Person(object):
num = 0
def __init__(self, name = None, age = None):
self.name = name
self.age = age
def eat(self):
print("eat food")
#定義一個(gè)實(shí)例方法
def run(self, speed):
print("%s在移動(dòng), 速度是 %d km/h"%(self.name, speed))
#定義一個(gè)類方法
@classmethod
def testClass(cls):
cls.num = 100
#定義一個(gè)靜態(tài)方法
@staticmethod
def testStatic():
print("---static method----")
#創(chuàng)建一個(gè)實(shí)例對(duì)象
P = Person("老王", 24)
#調(diào)用在class中的方法
P.eat()
#給這個(gè)對(duì)象添加實(shí)例方法
P.run = types.MethodType(run, P)
#調(diào)用實(shí)例方法
P.run(180)
#給Person類綁定類方法
Person.testClass = testClass
#調(diào)用類方法
print(Person.num)
Person.testClass()
print(Person.num)
#給Person類綁定靜態(tài)方法
Person.testStatic = testStatic
#調(diào)用靜態(tài)方法
Person.testStatic()
5. 運(yùn)行的過程中刪除屬性、方法
刪除的方法:
del 對(duì)象.屬性名
delattr(對(duì)象, "屬性名")
通過以上例子可以得出一個(gè)結(jié)論:相對(duì)于動(dòng)態(tài)語言,靜態(tài)語言具有嚴(yán)謹(jǐn)性!所以,玩動(dòng)態(tài)語言的時(shí)候,小心動(dòng)態(tài)的坑!
那么怎么避免這種情況呢? 請(qǐng)使用slots,