日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

屬性與 @property 方法讓你的python更高效

瀏覽:147日期:2022-07-10 16:15:35

一、用屬性替代 getter 或 setter 方法

以下代碼中包含手動實現(xiàn)的 getter(get_ohms) 和 setter(set_ohms) 方法:

class OldResistor(object): def __init__(self, ohms): self._ohms = ohms self.voltage = 0 self.current = 0 def get_ohms(self): return self._ohms def set_ohms(self, ohms): self._ohms = ohmsr0 = OldResistor(50e3)print(f’Before: {r0.get_ohms()}’)r0.set_ohms(10e3)print(f’After: {r0.get_ohms()}’)# => Before: 50000.0# => After: 10000.0

這些工具方法有助于定義類的接口,使得開發(fā)者可以方便地封裝功能、驗證用法并限定取值范圍。但是在 Python 語言中,應(yīng)盡量從簡單的 public 屬性寫起:

class Resistor(object): def __init__(self, ohms): self.ohms = ohms self.voltage = 0 self.current = 0r1 = Resistor(50e3)print(f’Before: {r1.ohms}’)r1.ohms = 10e3print(f’After: {r1.ohms}’)# => Before: 50000.0# => After: 10000.0

訪問實例的屬性則可以直接使用 instance.property 這樣的格式。

如果想在設(shè)置屬性的同時實現(xiàn)其他特殊的行為,如在對上述 Resistor 類的 voltage 屬性賦值時,需要同時修改其 current 屬性。可以借助 @property 裝飾器和 setter 方法實現(xiàn)此類需求:

from resistor import Resistorclass VoltageResistor(Resistor): def __init__(self, ohms): super().__init__(ohms) self._voltage = 0 @property def voltage(self): return self._voltage @voltage.setter def voltage(self, voltage): self._voltage = voltage self.current = self._voltage / self.ohmsr2 = VoltageResistor(1e3)print(f’Before: {r2.current} amps’)r2.voltage = 10print(f’After: {r2.current} amps’)Before: 0 ampsAfter: 0.01 amps

此時設(shè)置 voltage 屬性會執(zhí)行名為 voltage 的 setter 方法,更新當(dāng)前對象的 current 屬性,使得最終的電流值與電壓和電阻相匹配。

@property 的其他使用場景

屬性的 setter 方法里可以包含類型驗證和數(shù)值驗證的代碼:

from resistor import Resistorclass BoundedResistor(Resistor): def __init__(self, ohms): super().__init__(ohms) @property def ohms(self): return self._ohms @ohms.setter def ohms(self, ohms): if ohms <= 0: raise ValueError(’ohms must be > 0’) self._ohms = ohmsr3 = BoundedResistor(1e3)r3.ohms = -5# => ValueError: ohms must be > 0

甚至可以通過 @property 防止繼承自父類的屬性被修改:

from resistor import Resistorclass FixedResistance(Resistor): def __init__(self, ohms): super().__init__(ohms) @property def ohms(self): return self._ohms @ohms.setter def ohms(self, ohms): if hasattr(self, ’_ohms’): raise AttributeError('Can’t set attribute') self._ohms = ohmsr4 = FixedResistance(1e3)r4.ohms = 2e3# => AttributeError: Can’t set attribute

要點

優(yōu)先使用 public 屬性定義類的接口,不手動實現(xiàn) getter 或 setter 方法 在訪問屬性的同時需要表現(xiàn)某些特殊的行為(如類型檢查、限定取值)等,使用 @property @property 的使用需遵循 rule of least surprise 原則,避免不必要的副作用 緩慢或復(fù)雜的工作,應(yīng)放在普通方法中

二、需要復(fù)用的 @property 方法

對于如下需求:編寫一個 Homework 類,其成績屬性在被賦值時需要確保該值大于 0 且小于 100。借助 @property 方法實現(xiàn)起來非常簡單:

class Homework(object): def __init__(self): self._grade = 0 @property def grade(self): return self._grade @grade.setter def grade(self, value): if not (0 <= value <= 100): raise ValueError(’Grade must be between 0 and 100’) self._grade = valuegalileo = Homework()galileo.grade = 95print(galileo.grade)# => 95

假設(shè)上述驗證邏輯需要用在包含多個科目的考試成績上,每個科目都需要單獨計分。則 @property 方法及驗證代碼就要重復(fù)編寫多次,同時這種寫法也不夠通用。

采用 Python 的描述符可以更好地實現(xiàn)上述功能。在下面的代碼中,Exam 類將幾個 Grade 實例作為自己的類屬性,Grade 類則通過 __get__ 和 __set__ 方法實現(xiàn)了描述符協(xié)議。

class Grade(object): def __init__(self): self._value = 0 def __get__(self, instance, instance_type): return self._value def __set__(self, instance, value): if not (0 <= value <= 100): raise ValueError(’Grade must be between 0 and 100’) self._value = valueclass Exam(object): math_grade = Grade() science_grade = Grade()first_exam = Exam()first_exam.math_grade = 82first_exam.science_grade = 99print(’Math’, first_exam.math_grade)print(’Science’, first_exam.science_grade)second_exam = Exam()second_exam.science_grade = 75print(’Second exam science grade’, second_exam.science_grade, ’, right’)print(’First exam science grade’, first_exam.science_grade, ’, wrong’)# => Math 82# => Science 99# => Second exam science grade 75 , right# => First exam science grade 75 , wrong

在對 exam 實例的屬性進(jìn)行賦值操作時:

exam = Exam()exam.math_grade = 40

Python 會將其轉(zhuǎn)譯為如下代碼:

Exam.__dict__[’math_grade’].__set__(exam, 40)

而獲取屬性值的代碼:

print(exam.math_grade)

也會做如下轉(zhuǎn)譯:

print(Exam.__dict__[’math_grade’].__get__(exam, Exam))

但上述實現(xiàn)方法會導(dǎo)致不符合預(yù)期的行為。由于所有的 Exam 實例都會共享同一份 Grade 實例,在多個 Exam 實例上分別操作某一個屬性就會出現(xiàn)錯誤結(jié)果。

second_exam = Exam()second_exam.science_grade = 75print(’Second exam science grade’, second_exam.science_grade, ’, right’)print(’First exam science grade’, first_exam.science_grade, ’, wrong’)# => Second exam science grade 75 , right# => First exam science grade 75 , wrong

可以做出如下改動,將每個 Exam 實例所對應(yīng)的值依次記錄到 Grade 中,用字典結(jié)構(gòu)保存每個實例的狀態(tài):

class Grade(object): def __init__(self): self._values = {} def __get__(self, instance, instance_type): if instance is None: return self return self._values.get(instance, 0) def __set__(self, instance, value): if not (0 <= value <= 100): raise ValueError(’Grade must be between 0 and 100’) self._values[instance] = valueclass Exam(object): math_grade = Grade() writing_grade = Grade() science_grade = Grade()first_exam = Exam()first_exam.math_grade = 82second_exam = Exam()second_exam.math_grade = 75print(’First exam math grade’, first_exam.math_grade, ’, right’)print(’Second exam math grade’, second_exam.math_grade, ’, right’)# => First exam math grade 82 , right# => Second exam math grade 75 , right

還有另外一個問題是,在程序的生命周期內(nèi),對于傳給 __set__ 的每個 Exam 實例來說,_values 字典都會保存指向該實例的一份引用,導(dǎo)致該實例的引用計數(shù)無法降為 0 從而無法被 GC 回收。解決方法是將普通字典替換為 WeakKeyDictionary:

from weakref import WeakKeyDictionaryself._values = WeakKeyDictionary()

參考資料

Effective Python

以上就是屬性與 @property 方法讓你的python更高效的詳細(xì)內(nèi)容,更多關(guān)于python 屬性與 @property 方法的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Python 編程
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
久久久久午夜电影| 日韩视频在线一区二区三区| 青青草国产成人99久久| 最新国产精品视频| 亚洲黄色免费av| 午夜精品一区二区三区国产| 日韩午夜电影| 香蕉久久国产| 欧美日韩高清| 在线亚洲免费| 国产精品毛片视频| 色婷婷综合网| 欧美交a欧美精品喷水| 亚洲最新无码中文字幕久久 | 麻豆精品新av中文字幕| 神马久久午夜| 911亚洲精品| 亚洲一区网站| 国产精选在线| 日产精品一区二区| 香蕉国产精品| 精品国产乱码| 蜜臀av在线播放一区二区三区| 久久精品99久久久| 欧美三级精品| 日韩成人午夜精品| 99在线精品免费视频九九视 | 亚洲一区网站| 精品欧美一区二区三区在线观看| 六月天综合网| 99久久婷婷| 日韩一区二区三免费高清在线观看| 欧美极品中文字幕| 午夜av成人| 国产日韩一区二区三区在线 | 亚洲免费观看高清完整版在线观| 亚洲香蕉网站| 日韩深夜视频| 精品少妇一区| 麻豆传媒一区二区三区| 99久久夜色精品国产亚洲1000部| 国产一区二区久久久久| 国产精品伦一区二区| 成人在线网站| 欧美极品一区二区三区| 91精品日本| 国产日韩一区二区三区在线| 亚洲欧美日韩国产一区二区| 桃色一区二区| av免费不卡国产观看| 国产精品啊v在线| 高潮久久久久久久久久久久久久| 日本va欧美va欧美va精品| caoporn视频在线| 精品美女视频 | 九九综合九九| 一本色道精品久久一区二区三区| 国产精品88久久久久久| 91精品成人| 亚洲91在线| 精品视频黄色| 婷婷成人综合| 91成人精品观看| 国产a久久精品一区二区三区| 日韩欧美不卡| 激情综合网站| 亚州av日韩av| 加勒比视频一区| 国产精品麻豆成人av电影艾秋 | 亚洲特色特黄| 亚洲精品福利| 久久精品 人人爱| 97精品国产一区二区三区 | 麻豆久久精品| 日韩1区2区日韩1区2区| 精品无人区麻豆乱码久久久| 中文字幕在线官网| 伊人久久成人| 亚洲ab电影| 在线亚洲人成| 日韩一区二区三区四区五区| 高清不卡亚洲| 中文字幕日韩亚洲| 麻豆精品在线视频| 综合激情视频| 久久精品国产久精国产| 九色精品91| 老鸭窝亚洲一区二区三区| 性欧美xxxx免费岛国不卡电影| 日韩1区2区日韩1区2区| 久久久精品久久久久久96| 中文精品在线| 日本不卡视频一二三区| 麻豆视频一区二区| 亚洲综合日韩| av高清一区| 国产精品啊啊啊| 久久国产66| 日本欧美大码aⅴ在线播放| 久久激情综合网| 日本不卡一区二区三区| 亚洲国产专区| 性欧美69xoxoxoxo| 成人啊v在线| 亚洲伦乱视频| 精品久久电影| 国产精品黄网站| 中文不卡在线| 99国产精品| 伊人精品一区| 红桃视频欧美| 午夜国产欧美理论在线播放 | 久久久精品国产**网站| 亚洲国产成人精品女人| 中文字幕免费一区二区| 美女网站一区| 亚洲福利一区| 欧美不卡在线| 欧美中文一区二区| 天堂中文在线播放| 国产精品igao视频网网址不卡日韩| 亚洲综合图色| 免费看日韩精品| 美女精品网站| 亚洲精品福利| 欧美一级二区| 国产免费播放一区二区| 日韩激情视频网站| 亚洲精品自拍| 亚洲一区二区三区久久久| 亚洲色诱最新| 日韩一区二区三区免费视频| 婷婷成人av| 国产劲爆久久| 国产成人免费精品| 精品国产精品国产偷麻豆| 精品欧美视频| 亚洲福利精品| 在线免费观看亚洲| 日本视频一区二区| 成人午夜网址| 国产高清久久| 奇米色欧美一区二区三区| 精品黄色一级片| 国产在线看片免费视频在线观看| 99精品小视频| 久久国产精品免费一区二区三区 | 亚洲黄色中文字幕| 99成人在线视频| 蜜桃久久久久久久| 国产精品一区2区3区| 成人免费网站www网站高清| 亚洲免费影视| 久久中文字幕一区二区三区| 99久久久久国产精品| 日韩精品一区二区三区中文在线| 欧美交a欧美精品喷水| 激情综合网站| 麻豆精品视频在线观看免费| 亚洲精品91| 精品福利久久久| 性色一区二区| 国产精品二区不卡| 青青青免费在线视频| 日韩中文字幕区一区有砖一区| 国产午夜久久av| 久久激情一区| 亚洲精品欧美| 99久久99久久精品国产片果冰| 国产精品日本欧美一区二区三区| 国产精品日韩精品中文字幕| 人人精品亚洲| 视频在线观看91| 深夜福利视频一区二区| 免费成人在线观看| 欧美日韩色图| 91免费精品| 久久精品亚洲| 综合日韩在线| 久久xxxx| 怡红院精品视频在线观看极品| 女生影院久久| www.九色在线| 欧美一级鲁丝片| 日韩成人三级| 日韩国产欧美| 色婷婷精品视频| 在线手机中文字幕| 精品国产鲁一鲁****| 欧美激情麻豆| 精品国产亚洲一区二区在线观看| 91精品国产自产精品男人的天堂| 亚洲毛片在线| 视频一区二区欧美| 欧美日韩国产一区精品一区| 99久久夜色精品国产亚洲1000部| 日韩欧美看国产| 精品中文一区| 精品1区2区3区4区| 免费看的黄色欧美网站|