您当前的位置:首页 > IT编程 > python
| C语言 | Java | VB | VC | python | Android | TensorFlow | C++ | oracle | 学术与代码 | cnn卷积神经网络 | gnn | 图像修复 | Keras | 数据集 | Neo4j | 自然语言处理 | 深度学习 | 医学CAD | 医学影像 | 超参数 | pointnet | pytorch | 异常检测 | Transformers | 情感分类 | 知识图谱 |

自学教程:python property的使用技巧分享

51自学网 2021-10-30 22:41:30
  python
这篇教程python 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.__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     @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		# 就是起始值,即: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            @property    def 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    """    @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@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    @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.priceOut[23]: 1600.0In [24]: g.price = 3000In [25]: g.priceOut[25]: 2400.0In [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     16AttributeError: '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 = 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.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.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.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    @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.PRICEOut[60]: 7200.0In [61]: g.PRICE = 10000In [62]: g.PRICEOut[62]: 8000.0In [63]: del g.PRICE删除商品原价

可以发现两种都可以实现但 @property 装饰器的在 旧式类中只有 @property , 没有@method.setter 和

@method.deleter,新式类则两种都可以使用。因此看大家的习惯,选一种。

以上就是python property的使用技巧分享的详细内容,更多关于python property的资料请关注51zixue.net其它相关文章!


python中必会的四大高级数据类型(字符,元组,列表,字典)
详解python数据结构之队列Queue
万事OK自学网:51自学网_软件自学网_CAD自学网自学excel、自学PS、自学CAD、自学C语言、自学css3实例,是一个通过网络自主学习工作技能的自学平台,网友喜欢的软件自学网站。