Python 面向对象 之 Property

初识 @property

Property 是 Python 类的一个内置的 装饰器. @property 的作用是 将一个方法, 变为属性来调用.

装饰器: 在不改变原函数的基础上, 动态给函数增加功能

之前已经讨论过很多装饰器的细节了, 本质就是利用Pyhton变量的引用(指针)特性 和 变量命名空间来实现的.

def out(func):

    def inner(*args, **kwargs):

        print(f"...{args[0]} is checking...")
return func(*args, **kwargs) return inner @out
def check_2019_nCov(name):
msg = f"okay, {name} is very healthy."
return msg ret = check_2019_nCov("youge")
print(ret) # output
...youge is checking...
okay, youge is very health

@property 这个装饰器, 我一般会用在代码优化上多一点, 将一个方法, 封装为属性, 在阅读体验会好一些. 具体咋实现的, 也没搞清楚, 暂时能应用就好了.

具体写法即在需要的方法上边用 @property 装饰, 参数只有self, 调用的时候不加括号.

@property 将方法装饰为属性

# 访问类的私有属性
class Foo:
def __init__(self, score):
self.__score = score @property # 将方法装饰为属性
def show_score(self):
return self.__score if __name__ == '__main__':
s = Foo(66)
# 在写法上更加优雅, 不用多写 "方法()" 没用的括号
print(s.show_score) # output:
66

即在类自己的成员之间玩, (不需要传外部参数的情景下用, 表示 类的成员属性.), 初觉得比较 Pythonic 吧.

插一嘴, 上例中的 @property 所装饰的方法, 的功能是给 外部一个查看内部属性的接口. __ score 表示私有变量, 不能被直接访问, 期望上, 但 外部仍然是可以访问内部变量的, 感觉这算是 Pyhton不够严谨的地方吧, 没有绝对的安全哦.

s = Foo(66)

# 外部直接访问私有变量(方法) 是不可以的

print(s.__score)

# output
AttributeError: 'Foo' object has no attribute '__score'

but, 通过 实例对象.类名_ 私有属性 这样的方式是可以的 (为啥呢, 可以直接研究下Python类自身的构造原理)

s = Foo(66)

# 没有绝对的安全哦, 只要互相尊重
print(s._Foo__score) # output
66

@property 相关属性

通过上面的写法, 感觉无非就是, 把方法变成了属性而已, 似乎.除了写法上比较简洁优雅一点, 具体的使用场景是什么呢?

绑定属性-校验-手动实现

class Foo:
def __init__(self, score):
self.score = score # 绑定score属性
s = Foo(666)

更改 score 属性时, 我通常这样写.

s.score = 666

这样的无比粗暴 (直接将属性给暴露出去) 方式来 绑定属性 虽然写起来简单, 但是没有检验参数 导致可以随意被改.

这样粗暴的方法显然是不太合理的. 应该对 score进行一个校验, 值, 类型等都可以. 即我们在设置 score 属性的时候, 应该要实现两个 (自定义) 方法, get_score 用来查看值, 而 set_score 用来设置值.

class Foo:
def get_score(self):
"""获取属性值"""
print(self.score) def set_score(self, value):
"""设置属性值"""
if not isinstance(value, int):
raise ValueError("score 必须为整数哦")
if value > 100 or value <= 0:
raise ValueError("score 必须在 0-100哦") self.score = value if __name__ == '__main__':
f = Foo()
# 校验通过是可以的
f.set_score(66)
f.get_score() # output
66
# 随便设置 score 属性就不可以了.
if __name__ == '__main__':
f = Foo()
f.set_score('youge')
f.get_score() # output
raise ValueError("score 必须为整数哦")
ValueError: score 必须为整数哦
# 继续 debug
if __name__ == '__main__':
f = Foo()
f.set_score('youge')
f.get_score() # output
raise ValueError("score 必须在 0-100哦")
ValueError: score 必须在 0-100哦

case1 绑定属性-校验-@property 实现

  • 默认是 只读
  • setter
  • deleter
class Foo:

    @property  # read only
def score(self):
"""获取属性值"""
return self.score @score.setter
def score(self, value):
"""设置属性值"""
if not isinstance(value, int):
raise ValueError("score 必须为整数哦")
if value > 100 or value <= 0:
raise ValueError("score 必须在 0-100哦") # 这里一定不能 self.socre = value 哦, 不然就递归了.
self._score = value
print(f"set score = {value} successfully.") if __name__ == '__main__':
f = Foo()
f.score = 66 # print(f.score)
# output:
set score = 66 successfully
if __name__ == '__main__':
f = Foo()
f.score = 'youge' # output
raise ValueError("score 必须为整数哦")
ValueError: score 必须为整数哦

case 2 - 封装类属性

再来一小栗子吧. 嗯, 封装一个Book的类, 主要用来描述书籍的名称, 价格和评论. 然后呢, 要求能用 property 装饰器实现对 Book 的实例属性 set 和 get 方法进行定义. 即让实例对象可以用属性的方式来操作方法/属性.

class Book:
def __init__(self, name, price, comment=None):
self.__name = name
self.__price = price
self.__comment = comment # 传统方法对price属性进行 设置和访问
def get_price(self):
return self.__price def set_price(self, value):
self.__price = value # price = property(get_price, set_price) # @property 装饰器对 comment 属性进行 设置和访问
@property
def my_comment(self):
# 默认是 getter, 需要return 或 yeild
return self.__comment @my_comment.setter
def my_comment(self, value):
# 检验
if not isinstance(value, float):
raise ValueError("must be a decimal number") if value < 0 or value > 1000:
raise ValueError("0 ~ 1000") self.__comment = value @my_comment.deleter
def my_comment(self):
print("delete this atrr...")

小结

  • @property 装饰器的将方法变成属性来使用
  • @property 默认是只读, 配合 setter, deleter ... 等一起使用
  • 常用场景是对类实例属性赋值是, 进行一个 值的校验功能

Python 面向对象 之 @property的更多相关文章

  1. Python面向对象 -- slots, @property、多重继承MixIn、定制类(str, iter, getitem, getattr, call, callable函数,可调用对象)、元类(type, metaclass)

    面向对象设计中最基础的3个概念:数据封装.继承和多态 动态给class增加功能 正常情况下,当定义了一个class,然后创建了一个class的实例后,可以在程序运行的过程中给该实例绑定任何属性和方法, ...

  2. Python - 面向对象编程 - @property

    前言 前面讲到实例属性的时候,我们可以通过 实例对象.实例属性 来访问对应的实例属性 但这种做法是不建议的,因为它破坏了类的封装原则 正常情况下,实例属性应该是隐藏的,只允许通过类提供的方法来间接实现 ...

  3. python 面向对象七 property() 函数和@property 装饰符

    一.property引入 为了使对象的属性不暴露给调用者和进行属性值检查,设置了访问属性的接口函数,使用函数访问属性,并可以在函数内部检查属性. >>> class Student( ...

  4. python面向对象高级:@property

    @property 把方法『变成』了属性,广泛应用在类的定义中,可以让调用者写出简短的代码,同时保证对参数进行必要的检查,这样,程序运行时就减少了出错的可能性. 最大的作用就是既能检查参数,又可以用类 ...

  5. python面向对象进阶(八)

    上一篇<Python 面向对象初级(七)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...

  6. python 面向对象(进阶篇)

    上一篇<Python 面向对象(初级篇)>文章介绍了面向对象基本知识: 面向对象是一种编程方式,此编程方式的实现是基于对 类 和 对象 的使用 类 是一个模板,模板中包装了多个“函数”供使 ...

  7. python 面向对象和类成员和异常处理

    python 面向对象 你把自己想象成一个上帝,你要创造一个星球,首先你要把它揉成一个个球,两个直径就能创造一个球 class star: '''名字(name),赤道直径(equatorial di ...

  8. python 面向对象学习

    ------Python面向对象初 下面写一个类的简单实用,以便方便理解类 #python 3.5环境,解释器在linux需要改变 #阅读手册查询readme文件 #作者:S12-陈金彭 class ...

  9. python 面向对象、特殊方法与多范式、对象的属性及与其他语言的差异

    1.python 面向对象 文章内容摘自:http://www.cnblogs.com/vamei/archive/2012/06/02/2532018.html   1.__init__() 创建对 ...

  10. python面向对象(二)——类成员

    Python面向对象    类成员 1.字段         普通字段    属于对象         静态字段    属于类   2.方法 普通方法   触发者是对象    括号里至少一个参数 se ...

随机推荐

  1. NET中三种主机简单理解

    在NET中有三个不同的主机: .NET WebApplication 主机,也称为最小主机. 这是.NET 6中的一个新特性,旨在提供最小的启动时间和内存消耗.最小主机只包括.NET运行时的最基本组件 ...

  2. [ZJOI2019] 语言 题解

    不愧是 \(ZJOI\),<最可做的一道题>都让人一头雾水-- 首先将问题转化到链上. 可以将总共的组数转化为每个点可以到达的城市. 明显给每个点建一棵动态开点线段树,维护可以和他通商的点 ...

  3. 【COM3D2Mod 制作教程(5)】实战!制作身体部分(中)

    [COM3D2Mod 制作教程(5)]实战!制作身体部分(中) 帽子是很典型的装扮类型,较为简单适合入门,所以我们先制作帽子 Mod,流程基本和第二章中的概述相符.因为导入插件及其功能位置也都已在第二 ...

  4. Ansible - [06] Playbook

    Playbook 概述 Ansible ad-hoc 可以通过命令行形式远程管理其他主机 适合执行一些临时性简单任务 Ansible playbook 中文名称叫 剧本 将经常需要执行的任务写入一个文 ...

  5. springboot 中 java.util.zip 文件的压缩

    package com.geotmt.billingcenter.common.utils; import org.datanucleus.util.StringUtils; import org.s ...

  6. Python基础-模块和面向对象-shutil、re、bs4、requests模块

    概要: 模块 自定义模块(已经讲了) 内置模块 shutil re 正则表达式 第三方模块 requests 模块 bs4 模块 面向对象: 面向对象(Object-Oriented Programm ...

  7. 介绍一下opentcs

    OpenTCS是一个开源的自动运载系统(Automated Guided Vehicle,AGV)控制系统.它旨在管理和控制自动化运输车辆,例如AGV或自动搬运车(AMR),在工业和商业环境中执行各种 ...

  8. word 批量制作ppt

    将 Word 文档作为大纲,构建演示文稿 在 Word 文档中,单击"开始". 使用"样式"设置 Word 文档中内容的格式. 突出显示要用作幻灯片标题的内容, ...

  9. SpringSecurity学习笔记-前后端分离

    1. 简介 Spring Security是Spring家族中的一个安全管理框架.相比于另外一个安全框架Shiro,它提供了更丰富的功能,社区资源也比Shiro丰富. 一般来说中大型的项目都是使用Sp ...

  10. k8s v1.19版本之后,自签证书过期x509: certificate has expired or is not yet valid

    前言 在 Kubernetes 1.16 版本之前,kubeadm 工具的 alpha certs 子命令用于生成和管理 Kubernetes 集群的证书.然而,从 Kubernetes 1.19 版 ...