自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Python--- 對象的屬性

開發(fā) 后端
Python一切皆對象(object),每個對象都可能有多個屬性(attribute)。Python的屬性有一套統(tǒng)一的管理方案。

Python一切皆對象(object),每個對象都可能有多個屬性(attribute)。Python的屬性有一套統(tǒng)一的管理方案。

 

 

 

[[414572]]

 

屬性的__dict__系統(tǒng)

對象的屬性可能來自于其類定義,叫做類屬性(class attribute)。類屬性可能來自類定義自身,也可能根據(jù)類定義繼承來的。一個對象的屬性還可能是該對象實例定義的,叫做對象屬性(object attribute)。

對象的屬性儲存在對象的__dict__屬性中。__dict__為一個詞典,鍵為屬性名,對應(yīng)的值為屬性本身。我們看下面的類和對象。chicken類繼承自bird類,而summer為chicken類的一個對象。

 

  1. class bird(object): 
  2.     feather = True 
  3.  
  4. class chicken(bird): 
  5.     fly = False 
  6.     def __init__(self, age): 
  7.         self.age = age 
  8.  
  9. summer = chicken(2) 
  10.  
  11. print(bird.__dict__) 
  12. print(chicken.__dict__) 
  13. print(summer.__dict__) 

下面為我們的輸出結(jié)果:

  1. {'__dict__': <attribute '__dict__' of 'bird' objects>, '__module__''__main__''__weakref__': <attribute '__weakref__' of 'bird' objects>, 'feather'True'__doc__': None} 
  2.  
  3.  
  4. {'fly'False'__module__''__main__''__doc__': None, '__init__': <function __init__ at 0x2b91db476d70>} 
  5. {'age': 2} 

第一行為bird類的屬性,比如feather。第二行為chicken類的屬性,比如fly和__init__方法。第三行為summer對象的屬性,也就是age。有一些屬性,比如__doc__,并不是由我們定義的,而是由Python自動生成。此外,bird類也有父類,是object類(正如我們的bird定義,class bird(object))。這個object類是Python中所有類的父類。

可以看到,Python中的屬性是分層定義的,比如這里分為object/bird/chicken/summer這四層。當(dāng)我們需要調(diào)用某個屬性的時候,Python會一層層向上遍歷,直到找到那個屬性。(某個屬性可能出現(xiàn)在不同的層被重復(fù)定義,Python向上的過程中,會選取先遇到的那一個,也就是比較低層的屬性定義)。

當(dāng)我們有一個summer對象的時候,分別查詢summer對象、chicken類、bird類以及object類的屬性,就可以知道summer對象所有的__dict__,就可以找到通過對象summer可以調(diào)用和修改的所有屬性了。下面兩種屬性修改方法等效:

 

  1. summer.__dict__['age'] = 3 
  2. print(summer.__dict__['age']) 
  3.  
  4. summer.age = 5 
  5. print(summer.age) 

(上面的情況中,我們已經(jīng)知道了summer對象的類為chicken,而chicken類的父類為bird。如果只有一個對象,而不知道它的類以及其他信息的時候,我們可以利用__class__屬性找到對象的類,然后調(diào)用類的__base__屬性來查詢父類)

特性

同一個對象的不同屬性之間可能存在依賴關(guān)系。當(dāng)某個屬性被修改時,我們希望依賴于該屬性的其他屬性也同時變化。這時,我們不能通過__dict__的方式來靜態(tài)的儲存屬性。Python提供了多種即時生成屬性的方法。其中一種稱為特性(property)。特性是特殊的屬性。比如我們?yōu)閏hicken類增加一個特性adult。當(dāng)對象的age超過1時,adult為True;否則為False:

 

  1. class bird(object): 
  2.     feather = True 
  3.  
  4. class chicken(bird): 
  5.     fly = False 
  6.     def __init__(self, age): 
  7.         self.age = age 
  8.     def getAdult(self): 
  9.         if self.age > 1.0: return True 
  10.         elsereturn False 
  11.     adult = property(getAdult)   # property is built-in 
  12.  
  13. summer = chicken(2) 
  14.  
  15. print(summer.adult) 
  16. summer.age = 0.5 
  17. print(summer.adult) 

特性使用內(nèi)置函數(shù)property()來創(chuàng)建。property()最多可以加載四個參數(shù)。前三個參數(shù)為函數(shù),分別用于處理查詢特性、修改特性、刪除特性。最后一個參數(shù)為特性的文檔,可以為一個字符串,起說明作用。

我們使用下面一個例子進一步說明:

 

  1. class num(object): 
  2.     def __init__(self, value): 
  3.         self.value = value 
  4.     def getNeg(self): 
  5.         return -self.value 
  6.     def setNeg(self, value): 
  7.         self.value = -value 
  8.     def delNeg(self): 
  9.         print("value also deleted"
  10.         del self.value 
  11.     neg = property(getNeg, setNeg, delNeg, "I'm negative"
  12.  
  13. x = num(1.1) 
  14. print(x.neg) 
  15. x.neg = -22 
  16. print(x.value) 
  17. print(num.neg.__doc__) 
  18. del x.neg 

上面的num為一個數(shù)字,而neg為一個特性,用來表示數(shù)字的負數(shù)。當(dāng)一個數(shù)字確定的時候,它的負數(shù)總是確定的;而當(dāng)我們修改一個數(shù)的負數(shù)時,它本身的值也應(yīng)該變化。這兩點由getNeg和setNeg來實現(xiàn)。而delNeg表示的是,如果刪除特性neg,那么應(yīng)該執(zhí)行的操作是刪除屬性value。property()的最后一個參數(shù)("I'm negative")為特性negative的說明文檔。

使用特殊方法__getattr__

我們可以用__getattr__(self, name)來查詢即時生成的屬性。當(dāng)我們查詢一個屬性時,如果通過__dict__方法無法找到該屬性,那么Python會調(diào)用對象的__getattr__方法,來即時生成該屬性。比如:

 

  1. class bird(object): 
  2.     feather = True 
  3.  
  4. class chicken(bird): 
  5.     fly = False 
  6.     def __init__(self, age): 
  7.         self.age = age 
  8.     def __getattr__(self, name): 
  9.         if name == 'adult'
  10.             if self.age > 1.0: return True 
  11.             elsereturn False 
  12.         else: raise AttributeError(name
  13.  
  14. summer = chicken(2) 
  15.  
  16. print(summer.adult) 
  17. summer.age = 0.5 
  18. print(summer.adult) 
  19.  
  20. print(summer.male) 

每個特性需要有自己的處理函數(shù),而__getattr__可以將所有的即時生成屬性放在同一個函數(shù)中處理。__getattr__可以根據(jù)函數(shù)名區(qū)別處理不同的屬性。比如上面我們查詢屬性名male的時候,raise AttributeError。

(Python中還有一個__getattribute__特殊方法,用于查詢?nèi)我鈱傩?。__getattr__只能用來查詢不在__dict__系統(tǒng)中的屬性)

__setattr__(self, name, value)和__delattr__(self, name)可用于修改和刪除屬性。它們的應(yīng)用面更廣,可用于任意屬性。

即時生成屬性的其他方式

即時生成屬性還可以使用其他的方式,比如descriptor(descriptor類實際上是property()函數(shù)的底層,property()實際上創(chuàng)建了一個該類的對象)。有興趣可以進一步查閱。

總結(jié)

__dict__分層存儲屬性。每一層的__dict__只存儲該層新增的屬性。子類不需要重復(fù)存儲父類中的屬性。

即時生成屬性是值得了解的概念。在Python開發(fā)中,你有可能使用這種方法來更合理的管理對象的屬性。 

責(zé)任編輯:龐桂玉 來源: 馬哥Linux運維
相關(guān)推薦

2021-05-10 08:45:34

JavaScript開發(fā)對象

2011-07-04 17:27:42

JSP

2024-02-21 08:34:43

Figma圖形對象數(shù)據(jù)結(jié)構(gòu)

2024-05-13 08:39:18

Figma數(shù)據(jù)結(jié)構(gòu)編輯器

2009-07-29 15:07:23

Request對象的屬

2023-11-20 14:41:34

Python屬性

2023-12-01 10:20:04

Python類屬性

2009-12-24 16:46:42

ADO Connect

2023-05-16 07:35:29

2009-07-02 13:51:05

對象和范圍屬性

2009-10-22 10:10:20

VB.NET Proc

2010-09-28 10:33:59

HTML DOM Ch

2009-11-04 11:30:35

ADO.NET Dat

2021-03-26 23:41:19

JavaScript對象開發(fā)

2010-09-13 15:15:04

0級DOMDOM

2022-07-15 08:22:42

對象符串鍵Symbol

2023-06-02 15:42:51

JavaScript數(shù)據(jù)結(jié)構(gòu)對象

2011-03-31 15:40:43

ACCESS數(shù)據(jù)庫Fieldcaption屬性

2021-05-07 08:03:05

JS動態(tài)合并

2024-09-23 21:00:57

C#反射
點贊
收藏

51CTO技術(shù)棧公眾號