通常,當我們需要對物件的敏感屬性或者不希望外部直接訪問的屬性進行私有化,但是某些時候我們又需要對這些私有屬性進行修改,該怎麼處理呢?
1、幾個概念
_a(前置單下劃線),這種屬性僅表示約定的私有,非真正的私有。__a(前置雙下劃線),這種屬性表示私有,無法在外部訪問。__a__(前後雙下劃線),這種屬性標識系統屬性。(可選)a_(後置單下劃線),這種屬性是為了避免和保留關鍵字衝突。(可選)2、舉個例子
定義一個類:
class Student(object): _sex='male' __age=0
執行:(私有屬性無法在外部訪問)
>>> stu = Student()>>> stu._sex'male'>>> stu.__ageTraceback (most recent call last): File "<stdin>", line 1, in <module>AttributeError: 'Student' object has no attribute '__age'>>>
3、解決問題
從上面的類中我們可以看到,私有屬性無法在類例項中訪問,怎麼辦呢?當我們需要對類的私有屬性__age進行查詢和修改的時候,我們可以定義get_age和set_age去實現。
class Student(object): _sex='male' __age=0 def get_age(self): return self.__age def set_age(self,age): self.__age = age
執行:
>>> stu = Student()>>> stu.get_age() 0>>> stu.set_age(18) >>> stu.get_age() 18>>>
4、換個方法
但是上面的這種方式略顯複雜,如果在私有屬性較多的類中就不太適用了,所以我們期望尋求一種更簡單的方式去解決這個問題,比如將這個私有屬性轉化為另一個屬性。告訴你個好訊息,Python已經幫我們實現了,這就是@property。
class Student(object): _sex='male' __age=0 def get_age(self): return self.__age def set_age(self,age): self.__age = age @property def age(self): return self.__age
執行:
>>> from payhlib import Student>>> s = Student()>>> s.age0>>> s.set_age(19)>>> s.age19>>
在上面我們將__age私有屬性轉換為了age屬性,你可能會想,既然私有屬性轉換為了屬性,那我們是不是可以直接修改它呢?答案是不行,因為property雖然將__age轉換為了屬性,但是其不具備setter功能,需要我們去新增。
>>> from payhlib import Student>>> s = Student()>>> s.age 0>>> s.age=20Traceback (most recent call last): File "<stdin>", line 1, in <module>AttributeError: can't set attribute>>>
新增setter方法
class Student(object): _sex='male' __age=0 def get_age(self): return self.__age def set_age(self,age): self.__age = age @property def age(self): return self.__age @age.setter def age(self,value): self.__age=value
執行:
>>> from payhlib import Student>>> s = Student()>>> s.age0>>> s.age=20>>> s.age 20>>>
到此,@peoperty分享完畢,關於它的實現原理你可以檢視原始碼進行研究。
最新評論