Python中的属性操作(如获取、设置和删除属性)是我们日常编程中非常常见的操作。

但你有没有想过,当我们写下obj.attrobj.attr = value时,Python 内部究竟发生了什么?

本文将探讨Python属性的工作原理,并通过简单的代码示例来更好地理解这些概念。

1. 属性的基本操作

Python中,属性基本操作主要有三种:

  1. 获取属性值:value = obj.attr
  2. 设置属性值:obj.attr = value
  3. 删除属性: 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_gettp_descr_set两个函数指针实现。

3. 属性的存储与查找

Python中的属性存储在对象的字典(__dict__)中,或者在类的字典中。

当我们访问属性时,Python会按照以下顺序查找:

  1. 在实例的__dict__中查找
  2. 如果未找到,按照方法解析顺序在类及其父类的__dict__中查找
  3. 如果仍未找到,抛出AttributeError

比如:

class Parent:
x = "Parent" class Child(Parent):
y = "Child" obj = Child()
print(obj.x) # 输出 "Parent",从父类中找到
print(obj.y) # 输出 "Child",从子类中找到

4. 类型的属性管理

类型(如intlist)本身也有属性,这些属性的管理方式与普通对象类似,但有一些特殊之处。

因为类型本身是一个对象,它的类型是type,它是一个元类型

当我们访问类型属性时,Python会按照以下顺序查找:

  1. 在类型的字典中查找
  2. 如果未找到,按照 方法解析顺序 在元类型的字典中查找

这里有一个元类型的概念,所谓元类型,是指它的实例是一个类型

也就是说,正如用于创建对象一样,元类型是用于创建的。

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属性的工作原理的更多相关文章

  1. 『无为则无心』Python面向对象 — 51、私有成员变量(类中数据的封装)

    目录 1.私有成员变量介绍 (1)私有成员变量概念 (2)私有成员变量特点 (3)私有成员变量体验 2.属性私有化工作原理 3.定义成员变量的标识符规范 4.私有成员变量的获取和设置方式 1.私有成员 ...

  2. 2018-2019-2 20165316 『网络对抗技术』Exp3:免杀原理与实践

    2018-2019-2 20165316 『网络对抗技术』Exp3:免杀原理与实践 一 免杀原理与实践说明 (一).实验说明 任务一:正确使用msf编码器,msfvenom生成如jar之类的其他文件, ...

  3. 2017-2018-2 20155303 『网络对抗技术』Exp3:免杀原理与实践

    2017-2018-2 20155303 『网络对抗技术』Exp3:免杀原理与实践 --------CONTENTS-------- 1. 免杀原理与实践说明 实验说明 基础问题回答 2. 使用msf ...

  4. 『无为则无心』Python面向对象 — 60、魔法属性

    目录 1.魔法属性__name__ 2.魔法属性__bases__ 3.魔法属性__mro__ 4.魔法属性__doc__ 5.魔法属性__module__ 和__class__ 6.魔法属性__di ...

  5. 『无为则无心』Python面向对象 — 46、类和对象

    目录 1.理解类和对象 2.类 3.对象 4.Python中的对象 5.类和对象的定义 (1)定义类 (2)创建对象 (3)练习 6.拓展:isinstance() 函数 1.理解类和对象 (1)类和 ...

  6. 『无为则无心』Python基础 — 3、搭建Python开发环境

    目录 1.Python开发环境介绍 2.Python解释器的分类 3.下载Python解释器 4.安装Python解释器 5.Python解释器验证 1.Python开发环境介绍 所谓"工欲 ...

  7. 『无为则无心』Python基础 — 4、Python代码常用调试工具

    目录 1.Python的交互模式 2.IDLE工具使用说明 3.Sublime3工具的安装与配置 (1)Sublime3的安装 (2)Sublime3的配置 4.使用Sublime编写并调试Pytho ...

  8. 『无为则无心』Python函数 — 34、lambda表达式

    目录 1.lambda的应用场景 2.lambda语法 3.快速入门 4.示例:计算a + b 5.lambda的参数形式 6.lambda的应用 lambda表达式的主要作用就是化简代码. 匿名函数 ...

  9. python中的函数、生成器的工作原理

    1.python中函数的工作原理 def foo(): bar() def bar(): pass python的解释器,也就是python.exe(c编写)会用PyEval_EvalFramEx(c ...

  10. JVM 的 工作原理,层次结构 以及 GC工作原理

    JVM Java 虚拟机 Java 虚拟机(Java virtual machine,JVM)是运行 Java 程序必不可少的机制.JVM实现了Java语言最重要的特征:即平台无关性.原理:编译后的 ...

随机推荐

  1. 一款 IDEA 必备的 JSON 处理工具插件 — Json Assistant

    Json Assistant 是基于 IntelliJ IDEs 的 JSON 工具插件,让 JSON 处理变得更轻松! 主要功能 完全支持 JSON5 JSON 窗口(多选项卡) 选项卡更名 移动至 ...

  2. 基于云主机的ModelArts模型训练实践,让开发环境化繁为简

    本文分享自华为云社区<[开发者空间实践]云主机安装Docker并制作自定义镜像在ModelArts平台做模型训练>,作者: 开发者空间小蜜蜂. 1.1 案例介绍 在AI业务开发以及运行的过 ...

  3. 鸿蒙NEXT开发案例:九宫格随机

    [引言] 在鸿蒙NEXT开发中,九宫格抽奖是一个常见且有趣的应用场景.通过九宫格抽奖,用户可以随机获得不同奖品,增加互动性和趣味性.本文将介绍如何使用鸿蒙开发框架实现九宫格抽奖功能,并通过代码解析展示 ...

  4. 【Amadeus原创】SQL Server数据库备份、差异备份、日志备份脚本

    1,sp脚本 USE [master] GO /****** Object: StoredProcedure [dbo].[sp_BackupDatabase] Script Date: 2021/1 ...

  5. 【Amadeus原创】docker中修改wordpress上传文件大小

    1,进入docker 2,cp php.ini 3,   cd到目录 [root@3 ~]# docker exec -it wordpress /bin/bash root@1d8a4fbdaa6b ...

  6. 重磅宣布|强强联合,腾讯云携手Veeam提供云上数据存储服务

    近日获悉,腾讯云对象存储COS正式通过Veeam备份软件标准化测试,携手为用户提供云上数据存储服务. Veeam对COS的支持是通过SOBR( Scale out backup repository) ...

  7. win10 ocx控件注册失败的解决办法

    首先注意:ocx放在大部分系统目录无法注册,比如program file ,但windows目录可以. 第一步:关闭防火墙 第二步:确定是否缺少控件运行需要的微软运行库或其他运行环境 这里提供一个挺全 ...

  8. 【C#】【ffmpeg】外部调用线程执行ffmepg读取返回的信息乱码问题

    起因 C#使用FFmpeg获取电脑音视频可以用设备,当返回内容包含中文时,出现乱码问题 解决方案 ffmpeg本身的输出都是使用的错误输出,所以设置的是StandardErrorEncoding,如果 ...

  9. jenkins集群 - HTMLreport测试报告

    一.安装 HTML Publisher plugin 插件 插件下载地址 二.配置构建后步骤 三.编辑报告执行脚本 四.安装 Startup Trigger 和 Groovy 插件 1.安装原因: J ...

  10. SpringBoot:通过多个Context限制Bean的活动范围

    从Spring的包扫描说起 SpringBoot会扫描SpringBootApplication注解标注的类,他所在的包以及这个包的子包,把那里面的Bean注册到applicationContext中 ...