一、什么是 property特性

  • property装饰器用于将被装饰的方法伪装成一个数据属性,在使用时可以不用加括号而直接使用
  1. # ############### 定义 ###############
  2. class Foo:
  3. def func(self):
  4. pass
  5. # 定义property属性
  6. @property
  7. def prop(self):
  8. pass
  9. # ############### 调用 ###############
  10. foo_obj = Foo()
  11. foo_obj.func() # 调用实例方法
  12. foo_obj.prop # 调用property属性

如下的例子用于说明如何定一个简单的property属性:

  1. class Goods(object):
  2. @property
  3. def size(self):
  4. return 100
  5. g = Goods()
  6. print(g.size)
  1. 100

property属性的定义和调用要注意一下几点:

1. 定义时,在实例方法的基础上添加 @property 装饰器;并且仅有一个self参数

2. 调用时,无需括号

二、简单示例

对于京东商城中显示电脑主机的列表页面,每次请求不可能把数据库中的所有内容都显示到页面上,而是通过分页的功能局部显示,所以在向数据库中请求数据时就要显示的指定获取从第m条到第n条的所有数据 这个分页的功能包括:

  1. 根据用户请求的当前页和总数据条数计算出 m 和 n

  2. 根据m 和 n 去数据库中请求数据

  1. # ############### 定义 ###############
  2. class Pager:
  3. def __init__(self, current_page):
  4. # 用户当前请求的页码(第一页、第二页...)
  5. self.current_page = current_page
  6. # 每页默认显示10条数据
  7. self.per_items = 10
  8. @property
  9. def start(self):
  10. val = (self.current_page - 1) * self.per_items
  11. return val
  12. @property
  13. def end(self):
  14. val = self.current_page * self.per_items
  15. return val
  16. # ############### 调用 ###############
  17. p = Pager(1)
  18. print(p.start) # 就是起始值,即:m
  1. 0
  1. print(p.end) # 就是结束值,即:n
  1. 10

从上述可见Python的property属性的功能是:property属性内部进行一系列的逻辑计算,最终将计算结果返回。

三、 property属性的两种方式

  1. 装饰器 即:在方法上应用装饰器(推荐使用)

  2. 类属性 即:在类中定义值为property对象的类属性(Python2历史遗留)

3.1 装饰器

在类的实例方法上应用 @property 装饰器

Python中的类有经典类和新式类,新式类的属性比经典类的属性丰富。( 如果类继object,那么该类是新式类 )

经典类,具有一种 @property 装饰器:

  1. # ############### 定义 ###############
  2. class Goods:
  3. @property
  4. def price(self):
  5. return "laowang"
  6. # ############### 调用 ###############
  7. obj = Goods()
  8. result = obj.price # 自动执行 @property 修饰的 price 方法,并获取方法的返回值
  9. print(result)
  1. laowang

新式类,具有三种 @property 装饰器:

  1. #coding=utf-8
  2. # ############### 定义 ###############
  3. class Goods:
  4. """python3中默认继承object类
  5. 以python2、3执行此程序的结果不同,因为只有在python3中才有@xxx.setter @xxx.deleter
  6. """
  7. @property
  8. def price(self):
  9. print('@property')
  10. @price.setter
  11. def price(self, value):
  12. print('@price.setter')
  13. @price.deleter
  14. def price(self):
  15. print('@price.deleter')
  16. # ############### 调用 ###############
  17. obj = Goods()
  18. obj.price # 自动执行 @property 修饰的 price 方法,并获取方法的返回值
  1. @property
  1. obj.price = 123 # 自动执行 @price.setter 修饰的 price 方法,并将 123 赋值给方法的参数
  1. @price.setter
  1. del obj.price # 自动执行 @price.deleter 修饰的 price 方法
  1. @price.deleter

注意:

  • 经典类中的属性只有一种访问方式,其对应被 @property 修饰的方法

  • 新式类中的属性有三种访问方式,并分别对应了三个被 @property、@方法名.setter、@方法名.deleter 修饰的方法

由于新式类中具有三种访问方式,我们可以根据它们几个属性的访问特点,分别将三个方法定义为对同一个属性:获取、修改、删除

  1. class Goods(object):
  2. def __init__(self):
  3. # 原价
  4. self.original_price = 100
  5. # 折扣
  6. self.discount = 0.8
  7. @property
  8. def price(self):
  9. # 实际价格 = 原价 * 折扣
  10. new_price = self.original_price * self.discount
  11. return new_price
  12. @price.setter
  13. def price(self, value):
  14. self.original_price = value
  15. @price.deleter
  16. def price(self):
  17. print('del')
  18. del self.original_price
  19. obj = Goods()
  20. print(obj.price) # 获取商品价格
  1. 80.0
  1. obj.price = 200 # 修改商品原价
  2. print(obj.price)
  1. 160.0
  1. del obj.price # 删除商品原价
  1. del

3.2 类属性方式

创建值为property对象的类属性

注意:当使用类属性的方式创建property属性时,经典类和新式类无区别

  1. class Foo:
  2. def get_bar(self):
  3. return 'laowang'
  4. BAR = property(get_bar)
  5. obj = Foo()
  6. reuslt = obj.BAR # 自动调用get_bar方法,并获取方法的返回值
  7. print(reuslt)
  1. laowang

property方法中有个四个参数

  1. 第一个参数是方法名,调用 对象.属性 时自动触发执行方法

  2. 第二个参数是方法名,调用 对象.属性 = XXX 时自动触发执行方法

  3. 第三个参数是方法名,调用 del 对象.属性 时自动触发执行方法

  4. 第四个参数是字符串,调用 对象.属性.__doc__ ,此参数是该属性的描述信息

  1. #coding=utf-8
  2. class Foo(object):
  3. def get_bar(self):
  4. print("getter...")
  5. return 'laowang'
  6. def set_bar(self, value):
  7. """必须两个参数"""
  8. print("setter...")
  9. return 'set value' + value
  10. def del_bar(self):
  11. print("deleter...")
  12. return 'laowang'
  13. BAR = property(get_bar, set_bar, del_bar, "description...")
  14. obj = Foo()
  15. obj.BAR # 自动调用第一个参数中定义的方法:get_bar
  1. getter...
  2. 'laowang'
  1. obj.BAR = "alex" # 自动调用第二个参数中定义的方法:set_bar方法,并将“alex”当作参数传入
  1. setter...
  1. desc = Foo.BAR.__doc__ # 自动获取第四个参数中设置的值:description...
  2. print(desc)
  1. description...
  1. del obj.BAR # 自动调用第三个参数中定义的方法:del_bar方法
  1. deleter...

由于类属性方式创建property属性具有3种访问方式,我们可以根据它们几个属性的访问特点,分别将三个方法定义为对同一个属性:获取、修改、删除

  1. class Goods(object):
  2. def __init__(self):
  3. # 原价
  4. self.original_price = 100
  5. # 折扣
  6. self.discount = 0.8
  7. def get_price(self):
  8. # 实际价格 = 原价 * 折扣
  9. new_price = self.original_price * self.discount
  10. return new_price
  11. def set_price(self, value):
  12. self.original_price = value
  13. def del_price(self):
  14. del self.original_price
  15. PRICE = property(get_price, set_price, del_price, '价格属性描述...')
  16. obj = Goods()
  17. obj.PRICE # 获取商品价格
  1. 80.0
  1. obj.PRICE = 200 # 修改商品原价
  2. print(obj.PRICE)
  1. 160.0
  1. del obj.PRICE # 删除商品原价

综上所述:

  • 定义property属性共有两种方式,分别是【装饰器】和【类属性】,而【装饰器】方式针对经典类和新式类又有所不同。

  • 通过使用property属性,能够简化调用者在获取数据的流程

四、property+类的封装

  1. class People:
  2. def __init__(self, name):
  3. self.__name = name
  4. @property # 查看obj.name
  5. def name(self):
  6. return '<名字是:%s>' % self.__name
  7. peo1 = People('nick')
  1. print(peo1.name)
  1. <名字是:nick>
  1. try:
  2. peo1.name = 'EGON'
  3. except Exception as e:
  4. print(e)
  1. can't set attribute

五、应用

5.1 私有属性添加getter和setter方法

  1. class Money(object):
  2. def __init__(self):
  3. self.__money = 0
  4. def getMoney(self):
  5. return self.__money
  6. def setMoney(self, value):
  7. if isinstance(value, int):
  8. self.__money = value
  9. else:
  10. print("error:不是整型数字")

5.2 使用property升级getter和setter方法

  1. class Money(object):
  2. def __init__(self):
  3. self.__money = 0
  4. def getMoney(self):
  5. return self.__money
  6. def setMoney(self, value):
  7. if isinstance(value, int):
  8. self.__money = value
  9. else:
  10. print("error:不是整型数字")
  11. # 定义一个属性,当对这个money设置值时调用setMoney,当获取值时调用getMoney
  12. money = property(getMoney, setMoney)
  13. a = Money()
  14. a.money = 100 # 调用setMoney方法
  15. print(a.money) # 调用getMoney方法
  1. 100

5.3 使用property取代getter和setter方法

重新实现一个属性的设置和读取方法,可做边界判定

  1. class Money(object):
  2. def __init__(self):
  3. self.__money = 0
  4. # 使用装饰器对money进行装饰,那么会自动添加一个叫money的属性,当调用获取money的值时,调用装饰的方法
  5. @property
  6. def money(self):
  7. return self.__money
  8. # 使用装饰器对money进行装饰,当对money设置值时,调用装饰的方法
  9. @money.setter
  10. def money(self, value):
  11. if isinstance(value, int):
  12. self.__money = value
  13. else:
  14. print("error:不是整型数字")
  15. a = Money()
  16. a.money = 100
  17. print(a.money)
  1. 100

六、练习

计算圆的周长和面积

  1. import math
  2. class Circle:
  3. def __init__(self, radius): # 圆的半径radius
  4. self.radius = radius
  5. @property
  6. def area(self):
  7. return math.pi * self.radius**2 # 计算面积
  8. @property
  9. def perimeter(self):
  10. return 2 * math.pi * self.radius # 计算周长
  11. c = Circle(10)
  1. print(c.radius)
  1. 10
  1. print(c.area) # 可以向访问数据属性一样去访问area,会触发一个函数的执行,动态计算出一个值
  1. 314.1592653589793
  1. print(c.perimeter) # 同上
  1. 62.83185307179586

类的property特性的更多相关文章

  1. 类的封装,property特性,类与对象的绑定方法和非绑定方法,

    类的封装 就是把数据或者方法封装起来 为什么要封装 封装数据的主要原因是:保护隐私 封装方法的主要原因是:隔离复杂度(快门就是傻瓜相机为傻瓜们提供的方法,该方法将内部复杂的照相功能都隐藏起来了,比如你 ...

  2. day36 类的三大特性---封装以及Property特性

    目录 类的封装 如果真的要拿 类的property特性 setter & deleter 类属性用法 类与对象的绑定方法和非绑定方法 对象方法&类方法&静态方法 隐藏模块内的函 ...

  3. OC基础 类的三大特性

    OC基础  类的三大特性 OC的类和JAVA一样,都有三大特性:继承,封装,多态,那么我们就来看一下OC中类的三大特性. 1.继承 继承的特点: (1)子类从父类继承了属性和方法. (2)子类独有的属 ...

  4. Python深入浅出property特性属性

    导语 在Java中,通常在类中定义的成员变量为私有变量,在类的实例中不能直接通过对象.属性直接操作,而是要通过getter和setter来操作私有变量. 而在Python中,因为有property这个 ...

  5. day22-类的封装、property特性以及绑定方法与非绑定方法

    目录 类的封装 两个层面的封装 第一个层面 第二个层面 封装的好处 私有模块 类的propertry特性 setter 和 deleter 类与对象的绑定方法与非绑定方法 类的封装 将类的属性或方法隐 ...

  6. 第9章 Java类的三大特性之一:继承

    1.什么是继承 子类继承父类就是对父类的扩展,继承时会自动拥有父类所拥有的处private之外的所有成员作用:增加代码复用语法格式: class 子类名 extends 父类名{…………}第9章 Ja ...

  7. (转载)OC学习篇之---类的三大特性:封装,继承,多态

    之前的一片文章介绍了OC中类的初始化方法和点语法的使用,今天来继续学习OC中的类的三大特性,我们在学习Java的时候都知道,类有三大特性:继承,封装,多态,这个也是介绍类的时候,必须提到的话题,那么今 ...

  8. Java开发知识之Java类的高级特性,内部类.以及包使用.跟常量关键字

    Java开发知识之Java类的高级特性,内部类.以及包使用.跟常量关键字 一丶Java中包的机制 首先包其实就是个文件夹.作用就是管理类. Java中每次定义一个类的时候.通过Java编译之后.都会生 ...

  9. Python 类的三大特性的综合运用 案例

    # --------------------- 类的三大特性的综合运用 案例 ------------------------- # 定义三个类:小狗,小猫,人 # 小狗:姓名,年龄(默认1岁) 吃饭 ...

随机推荐

  1. 超实用的 Nginx 极简教程,覆盖了常用场景(转)

    概述 安装与使用 安装 使用 nginx 配置实战 http 反向代理配置 负载均衡配置 网站有多个 webapp 的配置 https 反向代理配置 静态站点配置 搭建文件服务器 跨域解决方案 参考 ...

  2. PAT 天梯赛 L1-050. 倒数第N个字符串 【字符串】

    题目链接 https://www.patest.cn/contests/gplt/L1-050 思路 因为是求倒数 我们不如直接 倒过来看 令 zzz 为第一个字符串 我们可以理解为 十进制 转换为 ...

  3. 写给精明Java开发者的测试技巧

    我们都会为我们的代码编写测试,不是吗?毫无疑问,我知道这个问题的答案可能会从 “当然,但你知道怎样才能避免写测试吗?” 到 “必须的!我爱测试”都有.接下来我会给你几个小建议,它们可以让你编写测试变得 ...

  4. FFmpeg音视频编解码实践总结

    PS:由于目前开发RTSP服务器传输模块时用到了h264文件,所以攻了一段时间去实现h264的视频编解码,借用FFmpeg SDK实现了任意文件格式之间的转换,并实现了流媒体实时播放,目前音视频同步需 ...

  5. iOS-个人开发者账号转公司开发者账号(邓白氏码申请教程)

    邓白氏编码申请 个人开发者账号转公司开发者账号,首先要申请邓白氏编码-DUNS,打开https://developer.apple.com/support/进行DUNS申请! 步骤如下: 1.选择Me ...

  6. C++软件工程师,你该会什么?

    请尊重原创: 转载注明来源   原创在这里哦 C语言广泛用于基础软件.桌面系统.网络通信.音频视频.游戏娱乐等诸多领域.是世界上使用最广泛的编程语言之一.随着物联网技术的发展,C/C++技术在3G网络 ...

  7. Centos 5.x 升级 python2.7,安装setuptools、mysqldb 完整记录

      最近由于有个工作任务需要搭个虚拟机环境,但是环境是搭建在内网,无法直接联网,很多软件都不能直接yum安装, 安装过程实在十分不顺利,在此留个记录给有需要的朋友. 环境是 CentOS 5.7 x6 ...

  8. 七号信令中TUP协议的主要消息和故障问题

      CIC码说明: TUP消息的路由标记: 为OPC 24位 DPC24位 CIC 12位,12位CIC 的低5为时隙号,其中后4位为SLC,高7位为系统号.     CIC配置需要注意几个问题:   ...

  9. Piggy-Bank(复习完全背包)

    传送门 题目大意: 有一个存钱的储存罐,给你它存满钱之前和之后的重量,和几类硬币的面值和重量. 求装满储钱罐时最小能得到多少钱. 题解:完全背包变形. 因为要求最小 一开始赋值大数. code: #i ...

  10. python装饰器执行顺序

    . python 装饰器 1) 2层装饰器 def decorator(func): # TODO def wrapper(*args, **kwargs): # TODO func(*args, * ...