0

0

如何理解Python的描述符(Descriptor)?

夜晨

夜晨

发布时间:2025-09-04 16:53:01

|

981人浏览过

|

来源于php中文网

原创

描述符通过实现__get__、__set__等方法控制属性访问,解决属性验证、计算等重复逻辑问题;数据描述符因实现__set__而优先级高于实例字典,非数据描述符则可被实例属性覆盖,这一机制支撑了property、方法绑定等核心功能;自定义如TypeValidator类可复用验证逻辑,利用__set_name__记录私有属性名,实现类型检查,提升代码声明性和维护性。

如何理解python的描述符(descriptor)?

Python的描述符(Descriptor)本质上是一种协议,它允许我们自定义一个对象在被作为另一个对象的属性访问时(获取、设置或删除)的行为。简单来说,当你访问

obj.attr
时,如果
attr
是一个实现了特定“描述符方法”(
__get__
__set__
__delete__
)的对象,Python就会把对
attr
的访问委托给这些方法来处理,而不是简单的字典查找。

描述符提供了一种强大且优雅的机制,用于控制类属性的访问逻辑。它不是什么魔法,而是Python对象模型中一个设计精巧的钩子,让你能在属性层面上注入自定义的行为。这就像给属性装了一个“守门员”,每次有人想碰这个属性,都得先经过守门员的检查或处理。

描述符究竟解决了什么问题?

说起来,描述符这东西,初看可能觉得有点抽象,但它解决的问题其实非常实际,而且你每天都在用,只是可能没意识到。最典型的例子就是

@property
装饰器,它就是描述符的一种应用。我们常常需要对属性进行验证、计算、懒加载或者干脆把它变成一个只读的“假属性”,以前可能得写一堆
get_foo()
set_foo()
方法,代码又臭又长。

描述符的出现,就是为了解决这种重复且分散的属性访问逻辑。它把这种逻辑集中封装在一个独立的、可重用的对象里。想象一下,你需要确保某个属性始终是正整数,或者它的值必须是字符串类型,又或者它的值是根据其他属性动态计算出来的。如果每次都在赋值前手动检查,那代码会变得非常冗余。描述符允许你把这些规则定义一次,然后像乐高积木一样,把它“插”到任何需要的属性上。

立即学习Python免费学习笔记(深入)”;

对我个人而言,描述符最吸引人的地方在于它提供了一种“声明式”的属性管理方式。你不再需要手动编写那些繁琐的

setter
getter
,而是通过定义一个描述符类,清晰地声明这个属性应该如何被访问和修改。这让代码更干净,意图更明确,也更容易维护。比如,在构建ORM(对象关系映射)时,每个数据库字段都可以是一个描述符,负责数据的序列化、反序列化以及与数据库的交互,这简直是天作之合。

数据描述符与非数据描述符有什么区别,为什么这很重要?

这是理解描述符的关键一环,也是很多人容易混淆的地方。Python在处理属性访问时,对这两种描述符的处理方式有着显著的不同。

  • 数据描述符(Data Descriptor):如果一个描述符同时实现了

    __set__
    方法(或者
    __delete__
    方法),那么它就是一个数据描述符。它的特点是,在属性查找时,数据描述符的优先级最高。这意味着,即使实例的
    __dict__
    中存在同名的键,Python也会优先调用数据描述符的
    __get__
    方法。它就像一个“铁面无私”的守卫,无论实例层面有没有同名属性,它都会拦截并处理。

  • 非数据描述符(Non-Data Descriptor):如果一个描述符只实现了

    __get__
    方法,而没有
    __set__
    __delete__
    ,那么它就是非数据描述符。它的特点是,实例的
    __dict__
    中的同名属性会覆盖非数据描述符
    。也就是说,如果
    obj.__dict__
    里已经有了
    attr
    这个键,那么访问
    obj.attr
    时就会直接返回
    obj.__dict__['attr']
    的值,而不会触发非数据描述符的
    __get__
    方法。只有当
    obj.__dict__
    中没有这个键时,Python才会去类中查找并调用非数据描述符。

    闪念贝壳
    闪念贝壳

    闪念贝壳是一款AI 驱动的智能语音笔记,随时随地用语音记录你的每一个想法。

    下载

为什么这个区别很重要?

这个区别直接影响了Python中很多核心机制的行为。最典型的就是方法(methods)。一个普通的函数,当它作为类属性被定义时,它其实就是一个非数据描述符。当你通过实例

obj.method()
调用它时,如果
obj.__dict__
里没有
method
这个键,Python会去类里找,发现
method
是一个函数(非数据描述符),就会调用它的
__get__
方法,把
obj
绑定为第一个参数(也就是
self
),然后返回这个绑定好的方法供你调用。但如果你给
obj.method
赋值了一个新值,比如
obj.method = 123
,那么
obj.__dict__
就会有一个
method
键,下次再访问
obj.method
时,就会直接返回
123
,而不会再调用原来的方法了。

理解这一点,能让你更好地把握Python的属性查找顺序,以及为什么有些时候你觉得属性行为“不符合预期”。它提供了一种灵活的机制,允许类级别的属性(比如方法)在实例层面被“个性化”甚至“覆盖”,同时又确保了像

@property
这种需要强制控制的属性不会被轻易绕过。

class DataDescriptor:
    def __get__(self, instance, owner):
        print(f"DataDescriptor __get__ called for {instance}")
        return instance._value if instance else "Class-level data"
    def __set__(self, instance, value):
        print(f"DataDescriptor __set__ called for {instance} with value {value}")
        instance._value = value # 通常将值存储在实例的私有属性中

class NonDataDescriptor:
    def __get__(self, instance, owner):
        print(f"NonDataDescriptor __get__ called for {instance}")
        return "Non-data value from descriptor"

class MyClass:
    data_attr = DataDescriptor()
    non_data_attr = NonDataDescriptor()

    def __init__(self, initial_value):
        self.data_attr = initial_value # 触发DataDescriptor.__set__
        self.instance_specific = "Instance specific value"

print("--- 访问数据描述符 ---")
obj = MyClass(100)
print(f"obj.data_attr: {obj.data_attr}") # 调用DataDescriptor.__get__
obj.data_attr = 200 # 再次调用DataDescriptor.__set__
print(f"obj.data_attr after update: {obj.data_attr}") # 再次调用DataDescriptor.__get__

# 即使我们在实例上设置了一个同名属性,数据描述符依然会优先
obj.__dict__['data_attr'] = "直接设置到实例字典"
print(f"obj.data_attr (after direct dict set): {obj.data_attr}") # 仍然是DataDescriptor.__get__被调用

print("\n--- 访问非数据描述符 ---")
print(f"obj.non_data_attr: {obj.non_data_attr}") # 调用NonDataDescriptor.__get__

# 现在,我们在实例上设置一个同名属性
obj.non_data_attr = "实例覆盖了描述符"
print(f"obj.non_data_attr (after instance override): {obj.non_data_attr}") # 直接从obj.__dict__获取,描述符不再被调用

print(f"MyClass.non_data_attr: {MyClass.non_data_attr}") # 通过类访问,描述符依然有效

这个例子清楚地展示了数据描述符如何“强制”其行为,而实例属性则可以轻易地“覆盖”非数据描述符。

如何自定义一个描述符,并用它实现一个简单的验证器?

自定义描述符通常涉及创建一个类,并在其中实现

__get__
__set__
__delete__
方法。这里我们以一个简单的类型验证器为例,它确保赋值给某个属性的值始终是预期的类型。

class TypeValidator:
    def __init__(self, expected_type):
        self.expected_type = expected_type
        self._private_name = None # 用于存储属性的实际名称

    def __set_name__(self, owner, name):
        """
        Python 3.6+ 引入的魔法方法,在描述符被赋值给类属性时调用。
        它允许描述符知道它被分配到的属性名称。
        """
        self._private_name = f'_{name}' # 用一个私有名称来存储实际值,避免与描述符自身冲突

    def __get__(self, instance, owner):
        if instance is None:
            return self # 通过类访问时,返回描述符自身
        # 从实例的字典中获取实际存储的值
        return instance.__dict__.get(self._private_name)

    def __set__(self, instance, value):
        if not isinstance(value, self.expected_type):
            raise TypeError(f"属性 '{self._private_name.lstrip('_')}' 期望类型为 {self.expected_type.__name__}, 但得到 {type(value).__name__}")
        # 将验证后的值存储到实例的字典中,使用私有名称
        instance.__dict__[self._private_name] = value

    def __delete__(self, instance):
        if self._private_name in instance.__dict__:
            del instance.__dict__[self._private_name]
        else:
            raise AttributeError(f"属性 '{self._private_name.lstrip('_')}' 未设置")

class User:
    name = TypeValidator(str)
    age = TypeValidator(int)
    email = TypeValidator(str) # 假设邮件也是字符串

    def __init__(self, name, age, email):
        self.name = name   # 触发 TypeValidator.__set__
        self.age = age     # 触发 TypeValidator.__set__
        self.email = email # 触发 TypeValidator.__set__

print("--- 创建一个有效的用户对象 ---")
user1 = User("Alice", 30, "alice@example.com")
print(f"用户1姓名: {user1.name}, 年龄: {user1.age}, 邮箱: {user1.email}")

print("\n--- 尝试设置一个无效类型的值 ---")
try:
    user1.age = "thirty" # 期望引发 TypeError
except TypeError as e:
    print(f"错误: {e}")

print("\n--- 再次检查用户1的年龄,确保未被修改 ---")
print(f"用户1年龄: {user1.age}")

print("\n--- 尝试删除属性 ---")
try:
    del user1.email
    print(f"用户1邮箱删除成功。现在访问: {user1.email}") # 应该返回None或引发AttributeError
except AttributeError as e:
    print(f"错误: {e}")

print("\n--- 通过类访问描述符 ---")
print(User.name) # 应该返回TypeValidator实例

在这个

TypeValidator
描述符中:

  1. __init__
    :初始化时接收期望的类型。
  2. __set_name__
    :这是一个非常实用的Python 3.6+特性。当
    TypeValidator
    实例被赋给
    User
    类的
    name
    age
    属性时,Python会自动调用这个方法,并把
    owner
    User
    类)和
    name
    "name"
    "age"
    )传进来。这让描述符可以知道它在类中被叫做什么,这对于存储实际值非常有用,我们用
    _private_name
    来构建一个唯一的键,避免与描述符自身冲突。
  3. __get__
    :当访问
    user1.name
    时,如果
    user1
    不是
    None
    (即通过实例访问),它会从
    user1.__dict__
    中查找我们用私有名称存储的实际值并返回。如果
    instance
    None
    (通过类访问,如
    User.name
    ),则返回描述符自身。
  4. __set__
    :当
    user1.name = "Bob"
    时,这个方法会被调用。它首先检查
    value
    是否是
    self.expected_type
    类型。如果不是,就抛出
    TypeError
    。如果是,它将值存储到
    instance.__dict__
    中,使用之前在
    __set_name__
    中确定的私有名称。
  5. __delete__
    :当
    del user1.name
    时调用,从
    instance.__dict__
    中删除对应的私有属性。

这种模式下,

TypeValidator
描述符本身作为类属性存在,所有
User
实例共享同一个
TypeValidator
实例。但每个
User
实例的实际
name
age
值,是存储在它们各自的
__dict__
中的,通过描述符的
__get__
__set__
方法进行间接访问和管理。这既保证了验证逻辑的复用,又确保了每个实例有自己独立的数据。

自定义描述符提供了一种非常灵活且强大的方式来控制属性行为,是Python进阶编程中一个不可或缺的工具。理解它,能让你对Python对象模型的理解更上一层楼。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

761

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

221

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1570

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

651

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1228

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1205

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

193

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

131

2025.08.07

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

26

2026.03.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.5万人学习

Django 教程
Django 教程

共28课时 | 5万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号