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的更多相关文章

  1. 「Fluent Python」今年最佳技术书籍

    Fluent Python 读书手记 Python数据模型:特殊方法用来给整个语言模型特殊使用,一致性体现.如:__len__, __getitem__ AOP: zope.inteface 列表推导 ...

  2. python property详解

    Python中有一个被称为属性函数(property)的小概念,它可以做一些有用的事情.在这篇文章中,我们将看到如何能做以下几点: 将类方法转换为只读属性 重新实现一个属性的setter和getter ...

  3. python property

    python property 在2.6版本中,添加了一种新的类成员函数的访问方式--property. 原型 class property([fget[, fset[, fdel[, doc]]]] ...

  4. python property装饰器

    直接上代码: #!/usr/bin/python #encoding=utf-8 """ @property 可以将python定义的函数“当做”属性访问,从而提供更加友 ...

  5. Python @property 详解

    本文讲解了 Python 的 property 特性,即一种符合 Python 哲学地设置 getter 和 setter 的方式. Python 有一个概念叫做 property,它能让你在 Pyt ...

  6. 学习笔记之Fluent Python

    Fluent Python by Luciano Ramalho https://learning.oreilly.com/library/view/fluent-python/97814919462 ...

  7. Python property() 函数

    Python property() 函数  Python 内置函数 描述 property() 函数的作用是在新式类中返回属性值. 语法 以下是 property() 方法的语法: class pro ...

  8. Fluent Python: memoryview

    关于Python的memoryview内置类,搜索国内网站相关博客后发现对其解释都很简单, 我觉得学习一个新的知识点一般都要弄清楚两点: 1, 什么时候使用?(也就是能解决什么问题) 2,如何使用? ...

  9. Python深入学习之《Fluent Python》 Part 1

    Python深入学习之<Fluent Python> Part 1 从上个周末开始看这本<流畅的蟒蛇>,技术是慢慢积累的,Python也是慢慢才能写得优雅(pythonic)的 ...

随机推荐

  1. GPUImage源码解读之GPUImageFramebuffer

    简介 OpenGL ES的FrameBuffer是渲染发生的地方,普通的2D图形的渲染默认发生在屏幕上:而三维的图形渲染则除了包括像素点的颜色,还有Depth Buffer,Stencil Buffe ...

  2. js常用共同方法

    var uh_rdsp = (function(){ //获取根目录 var getContextPath = function(){ var pathName = document.location ...

  3. Redis Sentinel 介绍

    Redis Sentinel   sentinel的功能: 监控:sentinel节点定期检测redis数据节点,其余sentinel节点是否可达. 通知:sentinel 节点会将故障转移结果通知给 ...

  4. 5. CSS是什么

    CSS概念 CSS,层叠样式表,也叫做风格样式表.通过CSS我们可以为页面添加一个美丽的外观,获得更加良好的用户体验.不过值得我们注意的是和HTML一样,CSS也不是编程语言,它只是提供一种配置文件, ...

  5. 虚拟机(unbutun16.04)设置静态ip

    电脑上装了虚拟机,想用xshell连接,无奈按照默认的网络设置方式每次重启了虚拟机后都要修改ip才能访问,这怎么能忍,经过一番折腾终于搞定这个问题了,解决步骤如下: 大步骤分为两步:其一是主机的设置, ...

  6. ES5拓展

    一.JSON拓展 1.JSON.parse(str,fun):将JSON字符串转为js对象 两个参数:str表示要处理的字符串:fun处理函数,函数有两个参数,属性名.属性值 // 定义json字符串 ...

  7. 『Linux基础 - 5 』Linux常用命令(2)

    这篇笔记的只要知识点: (1)ls查看文件信息,列表中每个字符所代表的含义 (2) 使用通配符匹配文件 (3) chmod命令:修改文件或目录权限 (4) 与用户相关命令(who.su.exit.pa ...

  8. vue实现图片路径传送

    <!DOCTYPE html><html lang="en"><head><meta charset="UTF-8"& ...

  9. Excelファイルを扱う方法

    概要 データをローカルに落としたいという要件がある場合.ユーザーはExcelを希望するケースが多いだろう.そんな時は以下の汎用モジュールを使用して簡単に作る事ができます.使用方法は.GUI_UPLOA ...

  10. java 关键字super和this

    super关键字 作用:调用父类的构造器 只能出现在子类的构造其中,并且必须是第一行 super()中的参数,决定了调用父类的那个构造器 注:如果子类构造器中没有出现super,则默认加上super( ...