飞翔飞翔
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
主页
  • 计算机基础

    • TCP/IP协议
    • Linux命令
  • 数据库

    • SQL教程
  • 编程语言

    • C语言
    • Python2
    • Python3
  • 数据格式

    • JSON教程
  • 工具

    • Markdown指南
  • Git

    • GitFlow
  • Quartz

    • Quartz教程
  • Java

    • Java设计模式
  • 缓存

    • Redis教程
联系
阿里云
  • 学习路径
  • 第1章 认识Python

    • Python 历史与特点
    • Python 2 与 Python 3 的核心差异
    • 安装与运行 Python 2.7.18
    • 编码规范 PEP 8
  • 第2章 基础语法

    • 变量与对象
    • 数字类型
    • 字符串 str
    • Unicode 字符串
    • 运算符
    • 空值 None
  • 第3章 流程控制

    • if 条件语句
    • if-else 条件语句
    • if-elif-else 多分支
    • 条件表达式(三元运算符)
    • while 循环
    • for 循环
    • range 与 xrange
    • 循环控制:break、continue、pass
    • 循环 else 子句
  • 第4章 数据结构

    • 列表基础
    • 列表方法
    • 列表推导式
    • 元组
    • 字典基础
    • 字典方法
    • 字典循环技巧
    • 集合
    • 序列解包
    • 序列比较
  • 第5章 函数

    • 定义函数
    • 参数传递机制
    • 默认参数
    • 关键字参数
    • 可变参数
    • Lambda 表达式
    • 文档字符串
    • 函数对象
  • 第6章 模块与包

    • import 导入
    • 模块搜索路径
    • name 与主程序
    • 编译文件 .pyc 与 .pyo
    • 包结构
    • dir() 函数
  • 第7章 文件与IO

    • 打开与关闭文件
    • 文件读写方法
    • with 上下文管理器
    • 格式化输出:% 操作符
    • 格式化输出:str.format()
    • JSON 序列化
  • 第8章 面向对象

    • 类定义与实例化
    • init 构造方法
    • 类变量与实例变量
    • 方法调用与 self
    • 继承基础
    • 多重继承
    • 新式类与旧式类
    • 私有变量与名称改写
    • 属性装饰器 property
    • 类方法与静态方法
    • 魔术方法
    • 空类与数据记录
  • 第9章 异常处理

    • 异常类型
    • try-except
    • try-except-else-finally
    • 抛出异常 raise
    • 自定义异常
    • with 语句与上下文管理器
  • 第10章 迭代器与生成器

    • 迭代器协议
    • 生成器函数
    • 生成器表达式
    • itertools模块
  • 第11章 标准库精要

    • os模块
    • sys模块
    • datetime模块
    • re模块
    • json模块
    • collections模块
    • math与random模块
    • urllib2与网络请求
    • subprocess与命令执行
    • threading与并发
    • unittest与测试
    • 虚拟环境与包管理
  • 第12章 工程实践

    • 调试技巧
    • 性能分析
    • 文档与注释
    • 下一步学习

属性装饰器 property

property 是 Python 中一种强大的机制,它把方法"伪装"成属性访问。通过 property,可以在读取、设置、删除属性时执行自定义逻辑,同时保持简洁的语法。

⚠️ property 只在新式类(继承 object)中工作。旧式类中 property 不会拦截属性访问。

基本用法

class Circle(object):
    def __init__(self, radius):
        self._radius = radius
    
    @property
    def radius(self):
        return self._radius
    
    @radius.setter
    def radius(self, value):
        if value < 0:
            raise ValueError("Radius cannot be negative")
        self._radius = value
    
    @property
    def area(self):
        return 3.14159 * self._radius ** 2

c = Circle(5)
print c.radius      # 5,像访问属性一样调用 getter
print c.area        # 78.53975

c.radius = 10       # 像赋值一样调用 setter
print c.area        # 314.159

c.radius = -1       # ValueError: Radius cannot be negative

@property 把 radius() 方法变成属性访问。@radius.setter 把 radius() 方法变成属性赋值。area 只有 getter,是只读属性。

等价的非装饰器写法

class Circle(object):
    def __init__(self, radius):
        self._radius = radius
    
    def get_radius(self):
        return self._radius
    
    def set_radius(self, value):
        if value < 0:
            raise ValueError("Radius cannot be negative")
        self._radius = value
    
    def get_area(self):
        return 3.14159 * self._radius ** 2
    
    radius = property(get_radius, set_radius)
    area = property(get_area)

property(fget, fset, fdel, doc) 是一个内置函数,接受 getter、setter、deleter 和文档字符串。装饰器语法只是它的语法糖。

实际应用场景

1. 数据验证:

class Temperature(object):
    def __init__(self, celsius=0):
        self._celsius = celsius
    
    @property
    def celsius(self):
        return self._celsius
    
    @celsius.setter
    def celsius(self, value):
        if value < -273.15:
            raise ValueError("Below absolute zero!")
        self._celsius = value
    
    @property
    def fahrenheit(self):
        return self._celsius * 9 / 5.0 + 32
    
    @fahrenheit.setter
    def fahrenheit(self, value):
        self.celsius = (value - 32) * 5 / 9.0

t = Temperature(25)
print t.fahrenheit      # 77.0
t.fahrenheit = 98.6
print t.celsius         # 37.0

2. 惰性计算:

class DataSet(object):
    def __init__(self, data):
        self._data = data
        self._stats = None
    
    @property
    def stats(self):
        if self._stats is None:
            self._stats = {
                "mean": sum(self._data) / float(len(self._data)),
                "max": max(self._data),
                "min": min(self._data),
            }
        return self._stats

ds = DataSet([1, 2, 3, 4, 5])
print ds.stats          # 第一次计算
print ds.stats          # 直接返回缓存结果

3. 向后兼容:

class Person(object):
    def __init__(self, first_name, last_name):
        self._first_name = first_name
        self._last_name = last_name
    
    @property
    def full_name(self):
        return "%s %s" % (self._first_name, self._last_name)
    
    @full_name.setter
    def full_name(self, value):
        parts = value.split()
        self._first_name = parts[0]
        self._last_name = parts[1] if len(parts) > 1 else ""

p = Person("Alice", "Smith")
print p.full_name       # Alice Smith
p.full_name = "Bob Jones"
print p._first_name     # Bob

删除属性

class ManagedAttribute(object):
    def __init__(self):
        self._value = None
    
    @property
    def value(self):
        if self._value is None:
            raise AttributeError("Value not set")
        return self._value
    
    @value.setter
    def value(self, val):
        self._value = val
    
    @value.deleter
    def value(self):
        self._value = None

obj = ManagedAttribute()
obj.value = 42
print obj.value         # 42
del obj.value           # 调用 deleter
print obj.value         # AttributeError: Value not set

与旧式类的对比

旧式类中 property 不工作:

class OldCircle:        # 旧式类!
    def __init__(self, radius):
        self._radius = radius
    
    @property
    def area(self):
        return 3.14 * self._radius ** 2

c = OldCircle(5)
print c.area            # <property object>,不是计算结果!

旧式类中 property 只是返回 property 对象本身,不会拦截属性访问。这是始终使用新式类的又一个理由。

上一页
私有变量与名称改写
下一页
类方法与静态方法