python property的使用技巧分享
作者:忆想不到的晖 发布时间:2022-11-18 21:25:51
目录
property属性
property属性的定义和调用要注意一下几点:
具体实例
property属性的有两种方式
装饰器方式
旧式类
新式类
注意
类属性方式
property对象与@property装饰器对比
property对象类属性
@property装饰器
property属性
一种用起来像是使用实例属性一样的特殊属性,可以对应于某个方法
既要保护类的封装特性,又要让开发者可以使用 对象.属性 的方式操作方法,@property 装饰器,可以直接通过方法名来访问方法,不需要在方法名后添加一对 () 小括号。
来看下求圆的面积的例子
class Circle(object):
PI = 3.14
def __init__(self, r):
# r圆的半径
self.r = r
self.__area = self.PI * self.r * self.r
@property
def area(self):
return self.__area
def get_area(self):
return self.__area
In [2]: c = Circle(10)
In [3]: c.area
Out[3]: 314.0
In [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
@property
def start(self):
val = (self.current_page - 1) * self.per_items
return val
@property
def end(self):
val = self.current_page * self.per_items
return val
# ipython测验
In [2]: p = Pager(1)
In [3]: p.start# 就是起始值,即:m
Out[3]: 0
In [4]: p.end# 就是结束值,即:n
Out[4]: 10
In [5]: p = Pager(2)
In [6]: p.start
Out[6]: 10
In [7]: p.end
Out[7]: 20
property属性的有两种方式
装饰器 即:在方法上应用装饰器 @property
类属性 即:在类中定义值为 property 对象的类属性 property()
装饰器方式
在类的实例方法上应用 @property 装饰器
Python中的类有旧式类 和 新式类,新式类 的属性比 旧式类的属性丰富。
旧式类
旧式类,具有一种 @property 装饰器
class Goods:
def __init__(self, name):
self.name = name
@property
def price(self):
return 100
# ipython测验
In [10]: g = Goods('手表')
In [11]: g.price
Out[11]: 100
新式类
新式类,具有三种 @property 装饰器
class Goods:
"""
python3中默认继承object类
以python2、3执行此程序的结果不同,因为只有在python3中才有@xxx.setter @xxx.deleter
"""
@property
def price(self):
print('@property')
@price.setter
def price(self, value):
print('@price.setter')
@price.deleter
def price(self):
print('@price.deleter')
# ipython测验
In [13]: g = Goods()
In [14]: g.price
@property
In [15]: g.price = 100
@price.setter
In [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
@property
def price(self):
# 实际价格 = 原价 * 折扣
new_price = self.original_price * self.discount
return new_price
@price.setter
def price(self, value):
self.original_price = value
@price.deleter
def price(self):
print('删除商品原价')
del self.original_price
# ipython测验
In [22]: g = Goods('小米手机', 2000)
In [23]: g.price
Out[23]: 1600.0
In [24]: g.price = 3000
In [25]: g.price
Out[25]: 2400.0
In [26]: del g.price
删除商品原价
In [27]: g.price
---------------------------------------------------------------------------
AttributeError Traceback (most recent call last)
<ipython-input-27-38ee45b469f2> in <module>
----> 1 g.price
<ipython-input-18-d5ea66eb7ece> in price(self)
12 def price(self):
13 # 实际价格 = 原价 * 折扣
---> 14 new_price = self.original_price * self.discount
15 return new_price
16
AttributeError: '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.BAR
Out[33]: 'get_bar'
f.BAR 自动调用 get_bar() 方法,并获取方法的返回值
property() 中有个四个参数
第一个参数是方法名,调用 对象.属性 时自动触发执行方法
第二个参数是方法名,调用 对象.属性 = XXX 时自动触发执行方法
第三个参数是方法名,调用 del 对象.属性 时自动触发执行方法
第四个参数是字符串,调用 对象.属性.__doc__ ,此参数是该属性的描述信息
class Foo(object):
def __init__(self, bar):
self.bar = bar
def get_bar(self):
print('get_bar')
return self.bar
def set_bar(self, value):
"""必须要有两个参数"""
print('set bar ' + value)
self.bar = value
def del_bar(self):
print('del bar')
del self.bar
BAR = property(get_bar, set_bar, del_bar, "bar description...")
# ipython测验
In [50]: f = Foo('python')
In [51]: f.BAR
get_bar
Out[51]: 'python'
In [52]: f.BAR = 'Java'
set bar Java
In [53]: f.BAR
get_bar
Out[53]: 'Java'
In [54]: del f.BAR
del bar
property对象与@property装饰器对比
由于 类属性方式 创建 property 对象属性具有3种访问方式,我们可以根据它们几个属性的访问特点,分别将三个方法定义为对 同一个属性:获取、修改、删除 ,跟 @property 装饰器对比。
property对象类属性
# Goods类 property对象类属性 应用
class Goods(object):
def __init__(self, name, price):
# 原价
self.original_price = price
# 折扣
self.discount = 0.8
def get_price(self):
# 实际价格 = 原价 * 折扣
new_price = self.original_price * self.discount
return new_price
def set_price(self, value):
self.original_price = value
def del_price(self):
print('删除商品原价')
del self.original_price
PRICE = property(get_price, set_price, del_price, "price description")
# ipython测验
In [59]: g = Goods('Mac电脑', 9000)
In [60]: g.PRICE
Out[60]: 7200.0
In [61]: g.PRICE = 10000
In [62]: g.PRICE
Out[62]: 8000.0
In [63]: del g.PRICE
删除商品原价
@property装饰器
# Goods类 @property装饰器 应用
class Goods(object):
def __init__(self, name, price):
# 原价
self.original_price = price
# 折扣
self.discount = 0.8
@property
def price(self):
# 实际价格 = 原价 * 折扣
new_price = self.original_price * self.discount
return new_price
@price.setter
def price(self, value):
self.original_price = value
@price.deleter
def price(self):
print('删除商品原价')
del self.original_price
# ipython测验
In [59]: g = Goods('Mac电脑', 9000)
In [60]: g.PRICE
Out[60]: 7200.0
In [61]: g.PRICE = 10000
In [62]: g.PRICE
Out[62]: 8000.0
In [63]: del g.PRICE
删除商品原价
可以发现两种都可以实现但 @property 装饰器的在 旧式类中只有 @property , 没有@method.setter 和
@method.deleter,新式类则两种都可以使用。因此看大家的习惯,选一种。
来源:https://cloud.tencent.com/developer/inventory/11966/article/1809087
猜你喜欢
- 1、在Python中以相对路径或者绝对路径来导入文件或者模块的方法今天在调试代码的时候,程序一直提示没有该模块,一直很纳闷,因为我导入文件一
- python给数据加上高斯噪声一开始用MATLAB给数据加噪声很简单,就一句话:% 给数据加指定SNR的高斯噪声signal_noise =
- 本博文的知识点一个是模块的调用和一个自定义函数返回值赋值给变量编写一个简单的函数模块:[root@bigdata zw]# more d.p
- 本文实例讲述了python自动化测试之setUp与tearDown的用法,分享给大家供大家参考。具体如下:实例代码如下:class Roma
- 在SQL Server中进行开发会让你身处险地,并且寻找快速解决方案。我们编辑了前十名关于SQL Server开发的常见问题。对常见的针对表
- Python np.argmin()和np.argmax()函数按照axis的要求返回最小的数/最大的数的下标numpy.argmin(a,
- 本文实例讲述了python处理二进制数据的方法。分享给大家供大家参考。具体如下:#!/usr/env/env python #-*- cod
- 一、默认参数python为了简化函数的调用,提供了默认参数机制:def pow(x, n = 2):r = 1 while n > 0
- 我们将研究一种判别式分类方法,其中直接学习评估 g(x)所需的 w 参数。我们将使用感知器学习算法。感知器学习算法很容易实现,但为了节省时间
- 在Oracle 8i版本之前,使用internal用户来执行数据库的启动和关闭以及create database等操作;从8i版本以后,Or
- Server对象主要是给编程人员提供一些方便的对象和属性。(1)ScriptTimeout属性:<%Server.ScriptTime
- 本文实例讲述了PHP解析xml格式数据工具类。分享给大家供大家参考,具体如下:class ome_xml { /**  
- 本文实例讲述了Python编程实现输入某年某月某日计算出这一天是该年第几天的方法。分享给大家供大家参考,具体如下:#基于 Pyth
- 导读:最近使用Thinkphp5.1做开发,在使用LOG_PATH常量(日志路径)时发生报错,因为之前一直使用5.0的框架,换到5.1版本后
- 网站内容的入口很大一部分都是依赖于导航系统,而网站的入口很大一部分依赖于搜索系统,这也在一定意义上证明了导航与搜索之间的重叠性。搜索系统可以
- 使用字符串创建矩阵是一个很实用的功能,之前自己尝试了很多次的小功能使用这个方法就能够简单实现。创建长度为16的字符串,是为了方便能够在各种数
- JavaScript中indexOf函数方法是返回 String 对象内第一次出现子字符串的字符位置。使用方法:strObj.indexOf
- 目录一、jieba库概述二、jieba库安装三、jieba分词的原理四、jieba分词的3种模式五、jieba库常用函数六、文本词频示例七、
- 目录创建conda环境下载Apple提供的tensorflow支持安装环境前准备工作安装依赖及相关库测试参考资料苹果为M1芯片的Mac提供了
- 由于不同的项目需要用不同的python版本,于是使用Anaconda来进行版本管理,现记录一下经验:在官网下载并安装好Anaconda以后(