Python進(jìn)階之@property動態(tài)屬性的實(shí)現(xiàn)
Python 動態(tài)屬性的概念可能會被面試問到,在項(xiàng)目當(dāng)中也非常實(shí)用,但是在一般的編程教程中不會提到,可以進(jìn)修一下。
先看一個簡單的例子。創(chuàng)建一個 Student 類,我希望通過實(shí)例來獲取每個學(xué)生的一些情況,包括名字,成績等。成績只有等到考試結(jié)束以后才會有,所以實(shí)例化的時候不會給它賦值。
class Student:
def __init__(self, name):
self.name = name
self.score = None
mike = Student('mike')
考試完以后,準(zhǔn)備給 mike 打分:
mike.score = 999
在這里,老師一不小心多打了個 9 ,通常來說打分都是 100 分值,999 是一個非法數(shù)據(jù),不應(yīng)該賦值成功。學(xué)生一多,老師打分出現(xiàn)手誤的情況肯定會越來越多,所以我們必須想辦法修改程序,限制 score 的值必須在 0-100 分。
限制值
我們定義一個方法,如果輸入的不是 0-100 的整數(shù),就讓程序報錯,數(shù)據(jù)合法,我們就把 score 屬性修改成功。
def set_score(self, new_score):
if not isinstance(new_score, int):
raise ValueError('score must be int')
if 0 <= new_score <= 100:
self.score = new_score
return self.score
else:
raise ValueError('score invalid')
這樣我們每次需要獲取成績的時候使用 self.score 獲取,修改成績的時候調(diào)用函數(shù)來修改:
mike.set_score(999)
調(diào)用以后會報錯,因?yàn)?999 是非法數(shù)據(jù)。注意,這個時候我使用 self.score 還是可以進(jìn)行設(shè)置,而且不報錯:
self.score = 999
這顯然是不行的。所以我們要提供一種機(jī)制,把 score 變成私有屬性,不能讓外部訪問。很遺憾,python 的私有屬性是偽私有。通常我們把 _ 開頭的屬性叫私有屬性,但是這只是一種協(xié)議和規(guī)定,你看到下劃線開頭的屬性,不要去訪問了。你硬要訪問,是可以的,python 并不會禁止。
使用 @property 的方式代替。
上面的方法雖然實(shí)現(xiàn)了功能,但是改變了屬性的使用方式。平常是這樣使用的:
# 獲取屬性
a = mike.score
# 設(shè)置屬性
mike.score = 99
@property
def score(self):
return self._score
@score.setter
def score(self, new_score):
if not isinstance(new_score, int):
raise ValueError('score must be int')
if 0 <= new_score <= 100:
self._score = new_score
return self._score
else:
raise ValueError('score invalid')
動態(tài)屬性的好處
- 統(tǒng)一了調(diào)用方式。self.score = 99 的方式,而不是函數(shù)調(diào)用的方式。
- _score 我們就不直接去使用了。你要用也可以,不建議。
- 如果我們一個屬性只可以讀,把 setter 部分注釋掉就可以了。
現(xiàn)在我們來完善這個類,添加 birth 屬性和年齡屬性:
from datetime import datetime
class Student:
def __init__(self, name, birth=1920):
self.name = name
self._score = None
self.birth = birth
self.age = datetime.now().year - self.birth
mike = Student('mike')
print(mike.birth)
print(mike.age)
birth 和 age 這兩個是可以根據(jù)一個求出另外一個的。存在數(shù)據(jù)冗余問題。
age 屬性這樣是有問題的。mike 初始化的時候,age 已經(jīng)被求出來了,如果我在下一年再去訪問 age 屬性,那他就是個錯誤的值。可以通過把 age 設(shè)成現(xiàn)在的秒數(shù)來驗(yàn)證:
self.age = datetime.now().second
mike = Student('mike')
time.sleep(5)
print(mike.age)
print(datetime.now().second)
動態(tài)顯示
@property def age(self): return datetime.now().year - self.birth
注意,這里不要去設(shè)置 @age.setter ,因?yàn)樗莿討B(tài)變化的,你修改了會造成數(shù)據(jù)不一致,它只能作為一個只讀屬性。
@property 作用和應(yīng)用場景:
- @property 優(yōu)化了屬性讀取和設(shè)置的可讀性
- 需要限制屬性的特征;
- 只讀屬性。如果屬性只可以讀,不可以寫,用起來很方便。
- 這個屬性根據(jù)一個變化的環(huán)境動態(tài)改變。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- python @property的用法及含義全面解析
- 介紹Python的@property裝飾器的用法
- 實(shí)例講解Python編程中@property裝飾器的用法
- python 中的@property的用法詳解
- Python 中@property的用法詳解
- Python @property裝飾器原理解析
- 詳解Python裝飾器之@property
- python裝飾器中@property屬性的使用解析
- Python @property原理解析和用法實(shí)例
- Python如何使用@property @x.setter及@x.deleter
- Python @property使用方法解析
- python中@Property屬性使用方法
- python中@property注解的具體使用
相關(guān)文章
python安裝numpy&安裝matplotlib& scipy的教程
下面小編就為大家?guī)硪黄猵ython安裝numpy&安裝matplotlib& scipy的教程。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-11-11
Python中輸入和輸出(打印)數(shù)據(jù)實(shí)例方法
在本篇文章里小編給大家整理的是關(guān)于Python中輸入和輸出(打印)數(shù)據(jù)實(shí)例方法以及相關(guān)知識點(diǎn),有需要的朋友們參考下。2019-10-10
python 使用遞歸的方式實(shí)現(xiàn)語義圖片分割功能
這篇文章主要介紹了python 使用遞歸的方式實(shí)現(xiàn)語義圖片分割,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-07-07
Python實(shí)現(xiàn)搜索算法的實(shí)例代碼
這篇文章主要介紹了Python實(shí)現(xiàn)搜索算法,文中通過實(shí)例代碼給大家分享高級搜索算法的想法,分為線性搜索和插值搜索,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下2020-01-01

