Python中关于property使用的小技巧
目录
- property属性
- 具体实例
- property属性的有两种方式
- 装饰器方式
- 旧式类
- 新式类
- 类属性方式
- property对象与@property装饰器对比
- property对象类属性
- @property装饰器
property属性 一种用起来像是使用实例属性一样的特殊属性,可以对应于某个方法
既要保护类的封装特性,又要让开发者可以使用 对象.属性 的方式操作方法,
@property 装饰器
,可以直接通过方法名来访问方法,不需要在方法名后添加一对 ()
小括号。来看下求圆的面积的例子
class Circle(object):PI = 3.14def __init__(self, r):# r圆的半径self.r = rself.__area = self.PI * self.r * self.r@propertydef area(self):return self.__areadef get_area(self):return self.__areaIn [2]: c = Circle(10)In [3]: c.areaOut[3]: 314.0In [4]: c.get_area()Out[4]: 314.0
property属性的定义和调用要注意一下几点:
- 定义时,在实例方法的基础上添加
@property
装饰器;并且仅有一个self
参数 - 调用时,无需括号
()
实例方法:c.get_area()property装饰的方法:c.area
具体实例 对于某商城中显示电脑主机的列表页面,每次请求不可能把数据库中的所有内容都显示到页面上,而是通过分页的功能局部显示,所以在向数据库中请求数据时就要显示的指定获取从第
m
条到第 n
条的所有数据 这个分页的功能包括:- 根据用户请求的当前页和总数据条数计算出 m 和 n
- 根据 m 和 n 去数据库中请求数据
class Pager(object):def __init__(self, current_page):# 用户当前请求的页码(第一页、第二页...)self.current_page = current_page# 每页默认显示10条数据self.per_items = 10 @propertydef start(self):val = (self.current_page - 1) * self.per_itemsreturn val@propertydef end(self):val = self.current_page * self.per_itemsreturn val# ipython测验In [2]: p = Pager(1)In [3]: p.start# 就是起始值,即:mOut[3]: 0In [4]: p.end# 就是结束值,即:nOut[4]: 10In [5]: p = Pager(2)In [6]: p.startOut[6]: 10In [7]: p.endOut[7]: 20
property属性的有两种方式
- 装饰器 即:在方法上应用装饰器
@property
- 类属性 即:在类中定义值为
property
对象的类属性property()
装饰器方式
在类的实例方法上应用
@property
装饰器Python中的类有
旧式类
和 新式类
,新式类
的属性比 旧式类
的属性丰富。旧式类 旧式类,具有一种
@property
装饰器class Goods:def __init__(self, name):self.name = name@propertydef price(self):return 100# ipython测验In [10]: g = Goods('手表')In [11]: g.priceOut[11]: 100
新式类 新式类,具有三种
@property
装饰器class Goods:"""python3中默认继承object类以python2、3执行此程序的结果不同,因为只有在python3中才有@xxx.setter@xxx.deleter"""@propertydef price(self):print('@property')@price.setterdef price(self, value):print('@price.setter')@price.deleterdef price(self):print('@price.deleter')# ipython测验In [13]: g = Goods()In [14]: g.price@propertyIn [15]: g.price = 100@price.setterIn [16]: del g.price@price.deleter
g.price
单独调用自动执行@property
修饰的price
方法,并获取方法的返回值g.price = 100
赋值自动执行@price.setter
修饰的price
方法,并将100
赋值给方法的参数del g.price
删除自动执行@price.deleter
修饰的price
方法
- 旧式类中的属性只有一种访问方式,其对应被
@property
修饰的方法 - 新式类中的属性有三种访问方式,并分别对应了三个被
@property
、@方法名.setter
、@方法名.deleter
修饰的方法
# Goods类@property应用class Goods(object):def __init__(self, name, price):# 原价self.original_price = price# 折扣self.discount = 0.8@propertydef price(self):# 实际价格 = 原价 * 折扣new_price = self.original_price * self.discountreturn new_price@price.setterdef price(self, value):self.original_price = value@price.deleterdef price(self):print('删除商品原价')del self.original_price# ipython测验In [22]: g = Goods('小米手机', 2000)In [23]: g.priceOut[23]: 1600.0In [24]: g.price = 3000In [25]: g.priceOut[25]: 2400.0In [26]: del g.price删除商品原价In [27]: g.price---------------------------------------------------------------------------AttributeErrorTraceback (most recent call last)in ----> 1 g.price in price(self)12def price(self):13# 实际价格 = 原价 * 折扣---> 14new_price = self.original_price * self.discount15return new_price16AttributeError: 'Goods' object has no attribute 'original_price'
类属性方式
创建值为
property
对象的类属性,当使用类属性的方式创建 property
属性时,旧式类
和 新式类
无区别class Foo:def get_bar(self):return 'get_bar'BAR = property(get_bar)# ipython 测验In [32]: f = Foo()In [33]: f.BAROut[33]: 'get_bar'
f.BAR
自动调用 get_bar()
方法,并获取方法的返回值property()
中有个四个参数- 第一个参数是方法名,调用 对象.属性 时自动触发执行方法
- 第二个参数是方法名,调用 对象.属性 = XXX 时自动触发执行方法
- 第三个参数是方法名,调用 del 对象.属性 时自动触发执行方法
- 第四个参数是字符串,调用 对象.属性._doc_ ,此参数是该属性的描述信息
class Foo(object):def __init__(self, bar):self.bar = bardef get_bar(self):print('get_bar')return self.bardef set_bar(self, value): """必须要有两个参数"""print('set bar ' + value)self.bar = valuedef del_bar(self):print('del bar')del self.barBAR = property(get_bar, set_bar, del_bar, "bar description...")# ipython测验In [50]: f = Foo('python')In [51]: f.BARget_barOut[51]: 'python'In [52]: f.BAR = 'Java'set bar JavaIn [53]: f.BARget_barOut[53]: 'Java'In [54]: del f.BARdel bar
property对象与@property装饰器对比 由于 类属性方式 创建
property
对象属性具有3种访问方式,我们可以根据它们几个属性的访问特点,分别将三个方法定义为对 同一个属性:获取、修改、删除 ,跟 @property
装饰器对比。property对象类属性
# Goods类 property对象类属性 应用class Goods(object):def __init__(self, name, price):# 原价self.original_price = price# 折扣self.discount = 0.8def get_price(self):# 实际价格 = 原价 * 折扣new_price = self.original_price * self.discountreturn new_pricedef set_price(self, value):self.original_price = valuedef del_price(self):print('删除商品原价')del self.original_pricePRICE = property(get_price, set_price, del_price, "price description")# ipython测验In [59]: g = Goods('Mac电脑', 9000)In [60]: g.PRICEOut[60]: 7200.0In [61]: g.PRICE = 10000In [62]: g.PRICEOut[62]: 8000.0In [63]: del g.PRICE删除商品原价
@property装饰器
# Goods类 @property装饰器 应用class Goods(object):def __init__(self, name, price):# 原价self.original_price = price# 折扣self.discount = 0.8@propertydef price(self):# 实际价格 = 原价 * 折扣new_price = self.original_price * self.discountreturn new_price@price.setterdef price(self, value):self.original_price = value@price.deleterdef price(self):print('删除商品原价')del self.original_price# ipython测验In [59]: g = Goods('Mac电脑', 9000)In [60]: g.PRICEOut[60]: 7200.0In [61]: g.PRICE = 10000In [62]: g.PRICEOut[62]: 8000.0In [63]: del g.PRICE删除商品原价
可以发现两种都可以实现但
@property
装饰器的在 旧式类中只有 @property
, 没有@method.setter
和@method.deleter
,新式类则两种都可以使用。因此看大家的习惯,选一种。大自然用数百亿年创造出我们现实世界,而程序员用几百年创造出一个完全不同的虚拟世界。我们用键盘敲出一砖一瓦,用大脑构建一切。人们把1000视为权威,我们反其道行之,捍卫1024的地位。我们不是键盘侠,我们只是平凡世界中不凡的缔造者 。
到此这篇关于Python中关于property使用的小技巧的文章就介绍到这了,更多相关Python property 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
推荐阅读
- 热闹中的孤独
- Shell-Bash变量与运算符
- JS中的各种宽高度定义及其应用
- 2021-02-17|2021-02-17 小儿按摩膻中穴-舒缓咳嗽
- 深入理解Go之generate
- 异地恋中,逐渐适应一个人到底意味着什么()
- 我眼中的佛系经纪人
- 《魔法科高中的劣等生》第26卷(Invasion篇)发售
- “成长”读书社群招募
- 2020-04-07vue中Axios的封装和API接口的管理