Fluent Python: @property
Fluent Python 9.6节讲到hashable Class,
为了使Vector2d类可散列,有以下条件:
(1)实现__hash__方法
(2)实现__eq__方法
(3)让Vector2d向量不可变
如何让Vector2d类实例的向量只读呢?可以使用property,如下所示:
class Vector2d:
def __init__(self, x, y):
self.__x = x
self.__y = y @property # The @property decorator marks the getter method of a property.
def x(self):
return self.__x @property # The @property decorator marks the getter method of a property.
def y(self):
return self.__y def __hash__(self):
return hash(self.__x) ^ hash(self.__y) def __eq__(self, other):
return hash(self) == hash(other) def __iter__(self):
return (i for i in (self.__x, self.__y))
现在我们在控制台尝试修改x或者y:
>>> import Example9_7
>>> v1 = Example9_7.Vector2d(3, 4)
>>> v1.x
3
>>> v1.x = 4
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: can't set attribute
>>> v1.y = 5
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
AttributeError: can't set attribute
这是我们想要的行为,但是为什么加上@properly装饰器后就变为只读了呢?我们需要对property有更深入的了解。
An Example To Begin With:
在深入了解property之前,我们先来看看property的应用场景:
假设我们写了一个关于温度的类:
class Celsius:
def __init__(self, temperature=0):
self.temperature = temperature def get_fahrenheit(self):
return self.temperature * 1.8 + 32
并且这个类渐渐变的很流行,被很多用户所调用,有一天,一个用户跑来建议说,温度不应该低于绝对温度-273摄氏度,他要求我们实现这个限制。
为了这样实现用户的要求,我们更新为v1.1:
class Celsius:
def __init__(self, temperature=0):
self.__temperature = temperature def get_fahrenheit(self):
return self.__temperature * 1.8 + 32 def get_temperature(self):
return self.__temperature def set_temperature(self, value):
if value < -273:
raise ValueError("Temperature below -273 is not possible.")
self.__temperature = value
用户的要求是实现了 可以这里有个问题,用户的代码里任然是这样获取温度的:
c = Celsius(37)
c.temperature = 20
current_temperature = c.temperature
而且代码里有成百上千行如此的代码,这些代码不得不改为:
c.set_temperature(20)
c.get_temperature()
对于用户来说这是很头疼的问题,因为我们的修改不是backward compatible.
The Power of @property:
对于这个问题,更为Pythonic的解决方式如下:
class Celsius:
def __init__(self, temperature=0):
self.__temperature = temperature def get_fahrenheit(self):
return self.__temperature * 1.8 + 32 def get_temperature(self):
return self.__temperature def set_temperature(self, value):
if value < -273:
raise ValueError("Temperature below -273 is not possible.")
self.__temperature = value temprature = property(get_temperature, set_temperature)
这样,用户任然像以前一样访问temprature:
>>> c1 = property_demo.Celsius(10)
>>> c1.temprature
10
>>> c1.temprature = 20
>>> c1.temprature
20
因此我们既实现了对termperature的限制,有保证了向后兼容
Digging Deeper into Property:
在Python里,property()是一个内建函数(Built-in function),它返回property 对象,函数原型是:
property(fget=None, fset=None, fdel=None, doc=None)
# fget is function to get value of the attribute, fset is function to set value of the attribute, fdel is function to delete the attribute and doc is a string (like a comment).
>>> property()
<property object at 0x7f50058bc5e8>
我们之前的示例可以分解为:
# make empty property
temperature = property()
# assign fget
temperature = temperature.getter(get_temperature)
# assign fset
temperature = temperature.setter(set_temperature)
我们也可以用装饰器来实现以上的功能:
class Celsius:
def __init__(self, temperature=0):
self.__temperature = temperature def get_fahrenheit(self):
return self.__temperature * 1.8 + 32 @property
def temperature(self):
return self.__temperature @temperature.setter
def temperature(self, value):
if value < -273:
raise ValueError("Temperature below -273 is not possible")
self.__temperature = value
装饰器版本是更为简单,推荐的方式。
Fluent Python: @property的更多相关文章
- 「Fluent Python」今年最佳技术书籍
Fluent Python 读书手记 Python数据模型:特殊方法用来给整个语言模型特殊使用,一致性体现.如:__len__, __getitem__ AOP: zope.inteface 列表推导 ...
- python property详解
Python中有一个被称为属性函数(property)的小概念,它可以做一些有用的事情.在这篇文章中,我们将看到如何能做以下几点: 将类方法转换为只读属性 重新实现一个属性的setter和getter ...
- python property
python property 在2.6版本中,添加了一种新的类成员函数的访问方式--property. 原型 class property([fget[, fset[, fdel[, doc]]]] ...
- python property装饰器
直接上代码: #!/usr/bin/python #encoding=utf-8 """ @property 可以将python定义的函数“当做”属性访问,从而提供更加友 ...
- Python @property 详解
本文讲解了 Python 的 property 特性,即一种符合 Python 哲学地设置 getter 和 setter 的方式. Python 有一个概念叫做 property,它能让你在 Pyt ...
- 学习笔记之Fluent Python
Fluent Python by Luciano Ramalho https://learning.oreilly.com/library/view/fluent-python/97814919462 ...
- Python property() 函数
Python property() 函数 Python 内置函数 描述 property() 函数的作用是在新式类中返回属性值. 语法 以下是 property() 方法的语法: class pro ...
- Fluent Python: memoryview
关于Python的memoryview内置类,搜索国内网站相关博客后发现对其解释都很简单, 我觉得学习一个新的知识点一般都要弄清楚两点: 1, 什么时候使用?(也就是能解决什么问题) 2,如何使用? ...
- Python深入学习之《Fluent Python》 Part 1
Python深入学习之<Fluent Python> Part 1 从上个周末开始看这本<流畅的蟒蛇>,技术是慢慢积累的,Python也是慢慢才能写得优雅(pythonic)的 ...
随机推荐
- 『ACM C++』 Codeforces | 1005D - Polycarp and Div 3
今天佛了,魔鬼周一,在线教学,有点小累,但还好,今天AC了一道,每日一道,还好达成目标,还以为今天完不成了,最近任务越来越多,如何高效完成该好好思考一下了~最重要的还是学业的复习和预习. 今日兴趣新闻 ...
- acm--1004
问题描述 再次比赛时间!看到气球在四周漂浮,多么兴奋.但要告诉你一个秘密,评委最喜欢的时间是猜测最流行的问题.比赛结束后,他们会统计每种颜色的气球并找出结果. 今年,他们决定离开这个可爱的工作给你. ...
- vue2.0 接收url参数
1) 路由配置传参方式在配置路由时 例如 "/firewall/authorize/:uid/:uname/:token"页面url为 http://XXX.com/firewal ...
- php连接数据库(一)
1.php链接数据库: 1.链接数据库 2.判断是否连接成功 3.设置字符集 4.选择数据库 5.准备SQL语句 6.发送SQL语句 7.处理结果集 8.释放资源(关闭数据库) $result = m ...
- php源码建博客5--建库建表-配置文件-错误日志
主要: 整理框架 建库建表 配置文件类 错误日志记录 --------------本篇后文件结构:-------------------------------------- blog ├─App │ ...
- Python3简单登录接口编写及遇到的问题分享
1.程序目标 输入用户名密码 认证成功后显示欢迎信息 输错三次后锁定 2.思路 利用python中的pickle模块,实现用户登录信息(采用dict)和被锁定用户信息(采用list)的存储.所以我预先 ...
- python学习笔记:第12天 列表推导式和生成器
目录 1. 迭代器 2. 推导式 1. 迭代器 什么是生成器呢,其实生成器的本质就是迭代器:在python中有3中方式来获取生成器(这里主要介绍前面2种) 通过生成器函数获取 通过各种推导式来实现生成 ...
- fiddler响应报文的headers属性详解
fiddler响应报文的headers属性详解 (1)Cache头域 1. Cache-Control 在请求报文已经说过了,用于设置缓存的属性,浏览内容不被缓存. 2. Data 生成消息的具体时间 ...
- 542. 01 Matrix
class Solution { public: vector<vector<int>> res; int m, n; vector<vector<int>& ...
- 用kubeadm构建k8s集群部署
一.环境 三台centos机器 二.软件及容器准备 1.安装docker环境 本例安装 docker-ce版本,repo源为docker-ce.repo文件,拷贝到 /etc/yum.repos.d下 ...