电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

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

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

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

以下代碼中包含手動實現的 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

這些工具方法有助于定義類的接口,使得開發者可以方便地封裝功能、驗證用法并限定取值范圍。但是在 Python 語言中,應盡量從簡單的 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 這樣的格式。

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

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

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

@property 的其他使用場景

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

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

要點

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

二、需要復用的 @property 方法

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

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

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

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

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 實例的屬性進行賦值操作時:

exam = Exam()exam.math_grade = 40

Python 會將其轉譯為如下代碼:

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

而獲取屬性值的代碼:

print(exam.math_grade)

也會做如下轉譯:

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

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

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 實例所對應的值依次記錄到 Grade 中,用字典結構保存每個實例的狀態:

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

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

from weakref import WeakKeyDictionaryself._values = WeakKeyDictionary()

參考資料

Effective Python

以上就是屬性與 @property 方法讓你的python更高效的詳細內容,更多關于python 屬性與 @property 方法的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
主站蜘蛛池模板: 变色龙云 - 打包app_原生app_在线制作平台_短链接_ip查询 | 不锈钢搅拌罐_高速搅拌罐厂家-无锡市凡格德化工装备科技有限公司 | 机械加工_绞车配件_立式离心机_减速机-洛阳三永机械厂 | 有机肥设备生产制造厂家,BB掺混肥搅拌机、复合肥设备生产线,有机肥料全部加工设备多少钱,对辊挤压造粒机,有机肥造粒设备 -- 郑州程翔重工机械有限公司 | 电车线(用于供电给电车的输电线路)-百科 | 西子馋火锅鸡加盟-太原市龙城酉鼎餐饮管理有限公司 | 全自动在线分板机_铣刀式在线分板机_曲线分板机_PCB分板机-东莞市亿协自动化设备有限公司 | 魔方网-培训咨询服务平台 | 水冷散热器_水冷电子散热器_大功率散热器_水冷板散热器厂家-河源市恒光辉散热器有限公司 | 算命免费_生辰八字_免费在线算命 - 卜算子算命网 | 斗式提升机,斗式提升机厂家-淄博宏建机械有限公司 | 高空重型升降平台_高空液压举升平台_高空作业平台_移动式升降机-河南华鹰机械设备有限公司 | 地磅-地秤-江阴/无锡地磅-江阴天亿计量设备有限公司_ | 英国雷迪地下管线探测仪-雷迪RD8100管线仪-多功能数字听漏仪-北京迪瑞进创科技有限公司 | CTAB,表面活性剂1631溴型(十六烷基三甲基溴化铵)-上海升纬化工原料有限公司 | 乐考网-银行从业_基金从业资格考试_初级/中级会计报名时间_中级经济师 | 干式变压器厂_干式变压器厂家_scb11/scb13/scb10/scb14/scb18干式变压器生产厂家-山东科锐变压器有限公司 | 福州甲醛检测-福建室内空气检测_环境检测_水质检测-福建中凯检测技术有限公司 | 模型公司_模型制作_沙盘模型报价-中国模型网 | 南京试剂|化学试剂|分析试剂|实验试剂|cas号查询-专业60年试剂销售企业 | 不锈钢管件(不锈钢弯头,不锈钢三通,不锈钢大小头),不锈钢法兰「厂家」-浙江志通管阀 | 间甲酚,间甲酚厂家-山东祥东新材料 | 美国HASKEL增压泵-伊莱科elettrotec流量开关-上海方未机械设备有限公司 | 小型UV打印机-UV平板打印机-大型uv打印机-UV打印机源头厂家 |松普集团 | 禹城彩钢厂_钢结构板房_彩钢复合板-禹城泰瑞彩钢复合板加工厂 | 钢绞线万能材料试验机-全自动恒应力两用机-混凝土恒应力压力试验机-北京科达京威科技发展有限公司 | 对辊式破碎机-对辊制砂机-双辊-双齿辊破碎机-巩义市裕顺机械制造有限公司 | 蒸压釜_蒸养釜_蒸压釜厂家-山东鑫泰鑫智能装备有限公司 | 屏蔽泵厂家,化工屏蔽泵_维修-淄博泵业 | 橡胶电子拉力机-塑料-微电脑电子拉力试验机厂家-江苏天源 | 学叉车培训|叉车证报名|叉车查询|叉车证怎么考-工程机械培训网 | COD分析仪|氨氮分析仪|总磷分析仪|总氮分析仪-圣湖Greatlake | 德州万泰装饰 - 万泰装饰装修设计软装家居馆| 雷达液位计_超声波风速风向仪_雨量传感器_辐射传感器-山东风途物联网 | 劳动法网-专业的劳动法和劳动争议仲裁服务网 | 合肥白癜风医院_[治疗白癜风]哪家好_合肥北大白癜风医院 | 硅胶制品-硅橡胶制品-东莞硅胶制品厂家-广东帝博科技有限公司 | 酒店厨房设计_中央厨房设计_北京商用厨房设计公司-奇能商厨 | T恤衫定做,企业文化衫制作订做,广告T恤POLO衫定制厂家[源头工厂]-【汉诚T恤定制网】 | 馋嘴餐饮网_餐饮加盟店火爆好项目_餐饮连锁品牌加盟指南创业平台 | 沈阳建筑设计公司_加固改造设计_厂房设计_设计资质加盟【金辉设计】 |