在繫結屬性時,如果我們直接把屬性暴露出去,雖然寫起來很簡單,但是,沒辦法檢查引數,導致可以把成績隨便改:
s =student()這顯然不合邏輯。為了限制score的範圍,可以通過乙個s.score = 9999
set_score()
方法來設定成績,再通過乙個get_score()
來獲取成績,這樣,在set_score()
方法裡,就可以檢查引數:
class現在,對任意的student例項進行操作,就不能隨心所欲地設定score了:student(object):
defget_score(self):
return
self._score
defset_score(self, value):
ifnot
isinstance(value, int):
raise valueerror('
score must be an integer!')
if value < 0 or value > 100:
raise valueerror('
score must between 0 ~ 100!')
self._score = value
>>> s =student()但是,上面的呼叫方法又略顯複雜,沒有直接用屬性這麼直接簡單。>>> s.set_score(60) #
ok!>>>s.get_score()
60>>> s.set_score(9999)
traceback (most recent call last):
...valueerror: score must between 0 ~ 100!
有沒有既能檢查引數,又可以用類似屬性這樣簡單的方式來訪問類的變數呢?對於追求完美的python程式設計師來說,這是必須要做到的!
還記得裝飾器(decorator)可以給函式動態加上功能嗎?對於類的方法,裝飾器一樣起作用。python內建的@property
裝飾器就是負責把乙個方法變成屬性呼叫的:
classstudent(object):
@property
defscore(self):
return
self._score
@score.setter
defscore(self, value):
ifnot
isinstance(value, int):
raise valueerror('
score must be an integer!')
if value < 0 or value > 100:
raise valueerror('
score must between 0 ~ 100!')
self._score = value
@property
的實現比較複雜,我們先考察如何使用。把乙個getter方法變成屬性,只需要加上@property
就可以了,此時,@property
本身又建立了另乙個裝飾器@score.setter
,負責把乙個setter方法變成屬性賦值,於是,我們就擁有乙個可控的屬性操作:
>>> s =student()注意到這個神奇的>>> s.score = 60 #
ok,實際轉化為s.set_score(60)
>>> s.score #
ok,實際轉化為s.get_score()
60>>> s.score = 9999traceback (most recent call last):
...valueerror: score must between 0 ~ 100!
@property
,我們在對例項屬性操作的時候,就知道該屬性很可能不是直接暴露的,而是通過getter和setter方法來實現的。
還可以定義唯讀屬性,只定義getter方法,不定義setter方法就是乙個唯讀屬性:
class上面的student(object):
@property
defbirth(self):
return
self._birth
@birth.setter
defbirth(self, value):
self._birth =value
@property
defage(self):
return 2015 - self._birth
birth
是可讀寫屬性,而age
就是乙個唯讀屬性,因為age
可以根據birth
和當前時間計算出來。
@property
廣泛應用在類的定義中,可以讓呼叫者寫出簡短的**,同時保證對引數進行必要的檢查,這樣,程式執行時就減少了出錯的可能性。
請利用@property
給乙個screen
物件加上width
和height
屬性,以及乙個唯讀屬性resolution
:
**:
#執行結果:練習:'''
請利用@property給乙個screen物件加上width和height屬性,
以及乙個唯讀屬性resolution
'''class
screen(object):
@property
defwidth(self):
return
self._width
@width.setter
defwidth(self,value):
ifnot
isinstance(value,int):
raise valueerror('
width must be an integer!')
elif value<=0:
raise valueerror('
width must greater than zero!')
self._width=value
@property
defheight(self):
return
self._height
@height.setter
defheight(self,value):
ifnot
isinstance(value,int):
raise valueerror('
height must be an integer!')
elif value<=0:
raise valueerror('
height must greater than zero!')
self._height=value
@property
defresolution(self):
return (self._width *self._height)
tv=screen()
tv.width=1024tv.height=768
print('
tv.width=
',tv.width)
print('
tv.height=
',tv.height)
try:
tv.resolution=10000
except
attributeerror as e:
print('
attributeerror:
',e)
print(tv.resolution) #
786432
assert tv.resolution == 786432, '
1024 * 768 = %d ?
' % tv.resolution
Vue中的計算屬性使用,動態賦值property
在使用vue時候,可能會發現乙個問題 我們不想在data中給某個property賦上固定的預設值,而是想賦值一些經過計算的property值,甚至使用到了其他property,第一時間可能會想到這麼寫,如下 錯誤示例 selectphotolabel this photolabels 0 text?...
Python基礎 使用 slots
定義乙個特殊的slots變數,來限制該class例項能新增的屬性 示例 usr bin env python3 coding utf 8 slots class student1 object pass defruntest1 s student1 s.name 蠟筆小新 s.age 6 s.sco...
python基礎 re使用
以乙個帶檔案操作的例子說明 目標 開啟乙個檔案,提取 中的內容,儲存在乙個新的檔案裡 涉及到的一些操作 序號操作 1檔案操作codecs.open 2正規表示式re模組的使用 上 import linecache import re import codecs filename lolstatist...