『Python底层原理』--Python属性的工作原理
Python中的属性操作(如获取、设置和删除属性)是我们日常编程中非常常见的操作。
但你有没有想过,当我们写下obj.attr或obj.attr = value时,Python 内部究竟发生了什么?
本文将探讨Python属性的工作原理,并通过简单的代码示例来更好地理解这些概念。
1. 属性的基本操作
在Python中,属性基本操作主要有三种:
- 获取属性值:
value = obj.attr - 设置属性值:
obj.attr = value - 删除属性:
del obj.attr
这些操作看似简单,但它们的实现却依赖于Python的底层机制。
为了更好地理解这些机制,我们需要先了解描述符的概念。
2. 描述符:属性的核心机制
描述符是Python中一个非常重要的概念,它是实现属性功能的核心机制。
描述符是一个对象,它通过实现特定的方法(如__get__、__set__和__delete__)来控制属性的访问、设置和删除。
例如,如下实现一个简单的描述符类:
class Descriptor:
def __get__(self, instance, owner):
return f"从 {instance} 获取属性值"
def __set__(self, instance, value):
print(f"设置 {instance} 的属性值: {value}")
def __delete__(self, instance):
print(f"从 {instance} 中删除属性值")
我们可以将这个描述符作为类属性使用:
obj = MyClass()
print(obj.attr) # 调用 __get__
obj.attr = "Hello" # 调用 __set__
del obj.attr # 调用 __delete__

在CPython中,描述符通过tp_descr_get和tp_descr_set两个函数指针实现。
3. 属性的存储与查找
Python中的属性存储在对象的字典(__dict__)中,或者在类的字典中。
当我们访问属性时,Python会按照以下顺序查找:
- 在实例的
__dict__中查找 - 如果未找到,按照方法解析顺序在类及其父类的
__dict__中查找 - 如果仍未找到,抛出
AttributeError
比如:
class Parent:
x = "Parent"
class Child(Parent):
y = "Child"
obj = Child()
print(obj.x) # 输出 "Parent",从父类中找到
print(obj.y) # 输出 "Child",从子类中找到
4. 类型的属性管理
类型(如int、list)本身也有属性,这些属性的管理方式与普通对象类似,但有一些特殊之处。
因为类型本身是一个对象,它的类型是type,它是一个元类型。
当我们访问类型属性时,Python会按照以下顺序查找:
- 在类型的字典中查找
- 如果未找到,按照 方法解析顺序 在元类型的字典中查找
这里有一个元类型的概念,所谓元类型,是指它的实例是一个类型。
也就是说,正如类用于创建对象一样,元类型是用于创建类的。
Python有一个内置的元类型,称为type,它是所有内置类型中的元类型。
比如我们在Python中创建一个类一般使用下面的方式:
class MyClass:
pass
也可以改成下面这样,和上面是等价的。
MyClass = type('MyClass', (), {})
5. 自定义属性管理
Python允许我们通过定义特殊方法来自定义属性的行为,这些方法包括:
__getattribute__:拦截所有属性访问__getattr__:仅在属性未找到时被调用__setattr__:拦截所有属性设置__delattr__:拦截所有属性删除
下面示例中,自定义了属性的访问和设置。
class MyClass:
def __getattribute__(self, name):
print(f"拦截所有属性的访问: {name}")
return super().__getattribute__(name)
def __getattr__(self, name):
print(f"拦截不存在属性的访问: {name}")
return f"属性 {name} 不存在"
def __setattr__(self, name, value):
print(f"设置属性值: {name} = {value}")
super().__setattr__(name, value)
obj = MyClass()
obj.x = "Hello"
print(obj.x) # 访问存在的属性
print(obj.y) # 访问不存在的属性

当设置属性(obj.x = "Hello")时,调用了 __setattr__;
访问存在的属性(print(obj.x))时,调用了__getattribute__;
访问不存在的属性(print(obj.y))时,调用了__getattribute__和__getattr__,
且是先调用__getattribute__。
6. 总结
Python的属性机制非常强大,它通过描述符、特殊方法和底层的字节码操作实现了灵活的属性管理。
通过理解这些机制,我们可以更好地设计类和对象,优化代码性能,并避免常见的陷阱。
『Python底层原理』--Python属性的工作原理的更多相关文章
- 『无为则无心』Python面向对象 — 51、私有成员变量(类中数据的封装)
目录 1.私有成员变量介绍 (1)私有成员变量概念 (2)私有成员变量特点 (3)私有成员变量体验 2.属性私有化工作原理 3.定义成员变量的标识符规范 4.私有成员变量的获取和设置方式 1.私有成员 ...
- 2018-2019-2 20165316 『网络对抗技术』Exp3:免杀原理与实践
2018-2019-2 20165316 『网络对抗技术』Exp3:免杀原理与实践 一 免杀原理与实践说明 (一).实验说明 任务一:正确使用msf编码器,msfvenom生成如jar之类的其他文件, ...
- 2017-2018-2 20155303 『网络对抗技术』Exp3:免杀原理与实践
2017-2018-2 20155303 『网络对抗技术』Exp3:免杀原理与实践 --------CONTENTS-------- 1. 免杀原理与实践说明 实验说明 基础问题回答 2. 使用msf ...
- 『无为则无心』Python面向对象 — 60、魔法属性
目录 1.魔法属性__name__ 2.魔法属性__bases__ 3.魔法属性__mro__ 4.魔法属性__doc__ 5.魔法属性__module__ 和__class__ 6.魔法属性__di ...
- 『无为则无心』Python面向对象 — 46、类和对象
目录 1.理解类和对象 2.类 3.对象 4.Python中的对象 5.类和对象的定义 (1)定义类 (2)创建对象 (3)练习 6.拓展:isinstance() 函数 1.理解类和对象 (1)类和 ...
- 『无为则无心』Python基础 — 3、搭建Python开发环境
目录 1.Python开发环境介绍 2.Python解释器的分类 3.下载Python解释器 4.安装Python解释器 5.Python解释器验证 1.Python开发环境介绍 所谓"工欲 ...
- 『无为则无心』Python基础 — 4、Python代码常用调试工具
目录 1.Python的交互模式 2.IDLE工具使用说明 3.Sublime3工具的安装与配置 (1)Sublime3的安装 (2)Sublime3的配置 4.使用Sublime编写并调试Pytho ...
- 『无为则无心』Python函数 — 34、lambda表达式
目录 1.lambda的应用场景 2.lambda语法 3.快速入门 4.示例:计算a + b 5.lambda的参数形式 6.lambda的应用 lambda表达式的主要作用就是化简代码. 匿名函数 ...
- python中的函数、生成器的工作原理
1.python中函数的工作原理 def foo(): bar() def bar(): pass python的解释器,也就是python.exe(c编写)会用PyEval_EvalFramEx(c ...
- JVM 的 工作原理,层次结构 以及 GC工作原理
JVM Java 虚拟机 Java 虚拟机(Java virtual machine,JVM)是运行 Java 程序必不可少的机制.JVM实现了Java语言最重要的特征:即平台无关性.原理:编译后的 ...
随机推荐
- JDK 18 及以上使用标准输出流中文输出乱码问题
著作权归作者所有. 商业转载请联系作者获得授权,非商业转载请注明出处. 链接:https://stazxr.cn/2024/12/05/JDK-18-以上使用标准输出流中文输出乱码问题/ 来源:終わり ...
- 特性Attribute的简单用法
一.建立一个自定义的Attribute类 注:类名+Attribute,类需要继承Attribute /// <summary> /// 特性 /// </summary> p ...
- Spring Boot + K8S 中的滚动发布、优雅停机、弹性伸缩、应用监控、配置分离
前言 K8s + SpringBoot实现零宕机发布:健康检查+滚动更新+优雅停机+弹性伸缩+Prometheus监控+配置分离(镜像复用) 配置 健康检查 健康检查类型:就绪探针(readiness ...
- Qt编写安防视频监控系统23-图片地图
一.前言 图片地图这个模块是后面新增加进去的,主要是安防领域还有很多应用场景是一个区域比如就一个学校,提供一个学校的平面图或者鸟瞰图,然后在该地图上放置对应的摄像机,双击该摄像机图标可以查看对应的实时 ...
- FluentAssertions:C#单元测试断言库,让测试代码更加直观、易读!
推荐一个C#开源库,用于单元测试中的断言,它提供了一系列的扩展方法,使得单元测试的断言看起来更加自然流畅. 01 项目简介 FluentAssertions 是一个基于 .NET 的断言库,它提供了一 ...
- UML之属性与参数的多重性
在UML中,多重性是指一个条目潜在的数量范围.多重性可被用于属性.操作参数.关联关系.UML元模型也使用多重性对元模型元素之间的关系进行约束.多重性总是包含基数值,它是相关条目在现实世界中的确切数量. ...
- minimind复现记录
- springcloud eureka原理和机制
公司的注册中心使用的是Eureka,之前使用过ZooKeeper,大致原理应该差不多,具体细节需要进一步学习,正好之前在腾讯云开发者社区看到一篇讲得很不错的文章,转载过来方便查看. 简介 在微服务架构 ...
- .Net Core 管道底层源码实现
在 .NET Core 中,请求处理管道是一个中间件(Middleware)链,用于处理 HTTP 请求并生成响应.管道的底层实现基于 Microsoft.AspNetCore.Http 命名空间中的 ...
- Python使用技巧2
python url网址拼接 在做爬虫中,经常会遇到需要把一个域名和网址路径进行拼接,在开发时一时没想到方法,特此记录下来. 利用parse.urljoin方法进行网址拼接 from urllib i ...