探索 Python 的 vars() 函数

        大家好,在软件开发的过程中,调试是一个不可或缺的环节。无论你是在解决 bug,优化代码,还是探索代码的执行流程,都需要一些有效的工具来帮助你更好地理解和调试代码。在 Python 编程中,vars() 函数是一个非常强大的工具,它可以让你在运行时动态地查看和修改对象的属性,为代码调试提供了便利。本文将深入探讨如何利用 vars() 函数的灵活性和功能,以提高代码调试的效率和准确性,希望能给大家带来一些帮助。

一、vars() 函数概述

   vars() 函数是 Python 的一个内置函数,它返回对象的 __dict__ 属性。__dict__ 属性是一个字典,包含了对象的所有属性和它们的值。对于类对象,__dict__ 属性包含了类的所有属性和方法。因此,vars() 函数可以用来获取对象的属性和属性值,或者类的属性和方法。       

使用格式:

vars(object)
  • object:要获取属性的对象。可以是模块、类、实例对象或其他具有 __dict__ 属性的对象。
返回值:
  • 如果 object 是模块,vars() 函数返回模块的 __dict__ 属性,其中包含了模块的所有全局变量和函数。
  • 如果 object 是类,vars() 函数返回类的 __dict__ 属性,其中包含了类的所有属性和方法。
  • 如果 object 是实例对象,vars() 函数返回实例对象的 __dict__ 属性,其中包含了实例对象的所有属性和它们的值。

示例:

class MyClass:class_variable = 10def __init__(self, a, b):self.a = aself.b = bobj = MyClass(1, 2)# 获取对象的属性和属性值
print(vars(obj))  # {'a': 1, 'b': 2}# 获取类的属性和方法
print(vars(MyClass))  # {'__module__': '__main__', 'class_variable': 10, '__init__': <function MyClass.__init__ at 0x7fe6e9f98670>, '__dict__': <attribute '__dict__' of 'MyClass' objects>, '__weakref__': <attribute '__weakref__' of 'MyClass' objects>, '__doc__': None}

         这就是 vars() 函数的基本概述,它是 Python 中一个强大且灵活的工具,可以用于获取对象的属性和属性值,或者类的属性和方法。

二、vars() 函数使用示例

1、在模块中使用 vars()

在模块中使用 vars() 函数可以获取模块的所有全局变量和函数。

# module.py
x = 10
y = 20def add(a, b):return a + bprint(vars())  # 获取模块的全局变量和函数

输出:

{'__name__': '__main__', '__doc__': None, '__package__': None, '__loader__': <_frozen_importlib_external.SourceFileLoader object at 0x7f80c405c550>, '__spec__': None, '__annotations__': {}, '__builtins__': <module 'builtins' (built-in)>, 'x': 10, 'y': 20, 'add': <function add at 0x7f80c405d310>}

2、在类中使用 vars()

在类中使用 vars() 函数可以获取类的所有属性和方法。

示例代码:

class MyClass:class_variable = 10def __init__(self, a, b):self.a = aself.b = bdef multiply(self, c):return self.a * self.b * cprint(vars(MyClass))  # 获取类的所有属性和方法

输出:

{'__module__': '__main__', 'class_variable': 10, '__init__': <function MyClass.__init__ at 0x7f80c405d0d0>, 'multiply': <function MyClass.multiply at 0x7f80c405d3a0>, '__dict__': <attribute '__dict__' of 'MyClass' objects>, '__weakref__': <attribute '__weakref__' of 'MyClass' objects>, '__doc__': None}

3、在实例对象中使用 vars()

在实例对象中使用 vars() 函数可以获取对象的所有属性和属性值。

示例代码:

class MyClass:def __init__(self, a, b):self.a = aself.b = bobj = MyClass(1, 2)
print(vars(obj))  # 获取对象的所有属性和属性值

输出:

{'a': 1, 'b': 2}

4、使用 vars() 动态添加对象属性

vars() 函数不仅可以获取对象的属性,还可以用于动态添加对象属性。

示例代码:

class MyClass:passobj = MyClass()# 使用 vars() 动态添加对象属性
vars(obj)['x'] = 10print(obj.x)  # 输出: 10

        在这个示例中,我们使用 vars(obj) 返回的字典来动态添加对象属性 x,然后可以直接通过 obj.x 来访问这个新添加的属性。

三、使用 vars() 函数的注意事项

当使用 vars() 函数时,有一些注意事项需要考虑:

可变性
  • vars() 函数返回的是对象的 __dict__ 属性,因此返回的是对象属性的引用。如果修改了返回的字典,会影响到对象的属性。
仅限具有 __dict__ 属性的对象
  • vars() 函数仅适用于具有 __dict__ 属性的对象。大多数类和实例都具有该属性,但也有一些特殊情况,比如使用 __slots__ 来限制对象属性的情况,这时可能不具有 __dict__ 属性。
调用方式
  • 如果不传递参数给 vars() 函数,它将返回当前作用域的 __dict__,即当前模块的全局变量和函数的字典。
安全性
  • 在某些情况下,可能不希望通过 vars() 来访问对象的属性,因为它可以访问到对象的私有属性。在代码设计中,应考虑对象属性的封装性,不要随意暴露对象的内部属性。

示例:

class MyClass:__private_attribute = "private"def __init__(self):self.public_attribute = "public"obj = MyClass()# 修改返回的字典,影响对象的属性
var_dict = vars(obj)
var_dict['public_attribute'] = "modified"
print(obj.public_attribute)  # 输出: modified# 尝试访问私有属性,成功访问到
print(var_dict['__private_attribute'])  # 输出: private# 如果不具有 __dict__ 属性,则会出错
try:vars(5)  # 无法获取整数对象的属性,会抛出 TypeError 异常
except TypeError as e:print(e)  # 输出: vars() argument must have __dict__ attribute# 通过 vars() 获取当前模块的全局变量和函数的字典
print(vars())  # 输出当前模块的全局变量和函数的字典

四、动态查看/创建对象属性

当需要在运行时动态查看对象的属性或者动态创建对象的属性时,vars() 函数可以派上用场。

1、动态查看对象属性

        使用 vars() 函数可以动态查看对象的属性。这对于在开发和调试过程中了解对象的内部状态非常有用。

示例:

class MyClass:def __init__(self, a, b):self.a = aself.b = bobj = MyClass(1, 2)# 动态查看对象的属性
print(vars(obj))  # 输出: {'a': 1, 'b': 2}

        在这个示例中,我们通过 vars() 函数动态查看了实例对象 obj 的属性,得到了一个包含对象属性和属性值的字典。

2、动态创建对象属性

        除了查看对象的属性外,vars() 函数还可以用于动态创建对象的属性。这在需要根据某些条件在运行时决定属性时非常有用。

示例:

class MyClass:passobj = MyClass()# 使用 vars() 动态添加对象属性
vars(obj)['x'] = 10print(obj.x)  # 输出: 10

        在这个示例中,我们通过 vars() 函数动态添加了对象属性 x,然后可以直接通过 obj.x 来访问这个新添加的属性。

五、vars()__slots__ 的关系

1、vars() 函数

  • vars() 函数是一个内置函数,用于返回对象的 __dict__ 属性,该属性是一个字典,包含对象的所有属性和它们的值。
  • 对于大多数对象,包括类实例对象和类对象,都有 __dict__ 属性,因此可以使用 vars() 函数来动态查看和修改对象的属性。

2、__slots__ 属性

  • __slots__ 是一个类级别的属性,用于限制类实例对象可以拥有的属性。它是一个列表,包含类实例对象允许定义的属性名称。
  • 当类定义了 __slots__ 属性时,类的实例对象将不再具有 __dict__ 属性,而是只能拥有 __slots__ 中指定的属性,这样可以节省内存空间。

3、关系:

  • 如果类定义了 __slots__ 属性,vars() 函数将无法使用,因为类的实例对象不再具有 __dict__ 属性。
  • 相反,如果类没有定义 __slots__ 属性,则可以使用 vars() 函数来动态查看和修改对象的属性。

示例:

class MyClassWithSlots:__slots__ = ['x', 'y']class MyClassWithoutSlots:passobj_with_slots = MyClassWithSlots()
obj_without_slots = MyClassWithoutSlots()# 对象属性的动态查看和修改
print(vars(obj_without_slots))  # 输出: {}
obj_without_slots.a = 10
print(vars(obj_without_slots))  # 输出: {'a': 10}# 尝试使用 vars() 函数访问具有 __slots__ 属性的对象
try:print(vars(obj_with_slots))  # 会抛出 AttributeError 异常
except AttributeError as e:print(e)  # 输出: 'MyClassWithSlots' object has no attribute '__dict__'

        在这个示例中,MyClassWithSlots 类定义了 __slots__ 属性,因此对象不再具有 __dict__ 属性,而 MyClassWithoutSlots 类没有定义 __slots__ 属性,因此对象仍然具有 __dict__ 属性,可以使用 vars() 函数查看和修改对象的属性。

六、使用 vars() 进行动态调试

动态查看对象属性
  • 在调试过程中,我们可以在代码中插入 print(vars(obj)) 语句来动态查看对象的属性和属性值。
  • 这可以帮助我们确认对象在某个特定点的状态是否符合预期,并且有助于我们定位代码中可能存在的问题。

示例:

class MyClass:def __init__(self, a, b):self.a = aself.b = bdef some_function(obj):print("Object state:", vars(obj))  # 在函数中动态查看对象属性obj = MyClass(1, 2)# 调用函数进行动态调试
some_function(obj)
动态修改对象属性
  • 在调试过程中,有时我们可能需要修改对象的属性以测试不同的情况或者修复问题。
  • 可以使用 vars() 函数获取对象的 __dict__ 属性,并修改其中的值来动态修改对象的属性。

示例:

class MyClass:def __init__(self, a, b):self.a = aself.b = bobj = MyClass(1, 2)# 动态修改对象属性
var_dict = vars(obj)
var_dict['a'] = 10
var_dict['b'] = 20print(vars(obj))  # 输出: {'a': 10, 'b': 20}
注意事项
  • 在生产环境中,不建议在代码中保留类似于 print(vars(obj)) 这样的调试语句,因为这会影响代码的性能。
  • 调试完成后,应该删除或者注释掉这些调试语句,以确保代码的性能和可维护性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/338991.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

国产可视化爬虫助力AI大模型训练:精准爬取汉语词典

大语言模型&#xff0c;可以生成流畅对话的会话聊天机器人、通畅起草文章的内容生成器。在炫酷技术的背后&#xff0c;数据、算力、算法&#xff0c;被视作生成式AI的三个核心要素。由此可见&#xff0c;高质量的训练数据对于AI算法的准确性至关重要。 如何获得高质量的训练数…

【嵌入式硬件】DRV8874电机驱动

目录 1 芯片介绍 1.1 特性简介 1.2 引脚配置 1.3 最佳运行条件 2 详细说明 2.1 PMODE配置控制模式 2.1.1 PH/EN 控制模式 2.1.2 PWM 控制模式 2.1.3 独立半桥控制模式 2.2 电流感测和调节 2.2.1 IPROPI电流感测 2.2.2 IMODE电流调节 3.应用 3.1设计要求 3.2 设计…

数据结构严蔚敏版精简版-绪论

1.基本概念和术语 下列概念和术语将在以后各章节中多次出现&#xff0c;本节先对这些概念和术语赋予确定的含义。 数据(Data)&#xff1a;数据是客观事物的符号表示&#xff0c;是所有能输入到计算机中并被计算机程序处理的符号 的总称。 数据元素(DataElement)&#xff1a;…

JVM运行时数据区 - 程序计数器

运行时数据区 Java虚拟机在执行Java程序的过程中&#xff0c;会把它管理的内存划分成若干个不同的区域&#xff0c;这些区域有各自的用途、创建及销毁时间&#xff0c;有些区域随着虚拟机的启动一直存在&#xff0c;有些区域则随着用户线程的启动和结束而建立和销毁&#xff0…

JAVAEE1

Web前端&#xff1a; 1.建立web开发的息维模式写代码不仅仅是为了实现某个功能&#xff0c;更是学习解决问题的思维方式 2.先使用&#xff0c;再理解&#xff0c;会导致刚开始比较懵&#xff0c;不知其所以然.切忌不可深陷其中&#xff0c; 3.涉及简单的软件工程的设计思想&…

Java Agent利器

一、JavaAgent技术 1.1 什么是JavaAgent JavaAgent是一种特殊的Java程序&#xff0c;是Instrumentation的客户端。它与普通Java程序通过main方法启动不同&#xff0c;JavaAgent并不是一个可以单独启动的程序&#xff0c;它必须依附在一个Java应用程序&#xff08;JVM&#xf…

Spring创建对象的多种方式

一、对象分类 简单对象&#xff1a;使用new Obj()方式创建的对象 复杂对象&#xff1a;无法使用new Obj()方式创建的对象。例如&#xff1a; 1. AOP创建代理对象。ProxyFactoryBean; 2. Mybatis中的SqlSessionFactoryBean; 3. Hibernate中的SessionFactoryBean。二、创建对象方…

Docker学习笔记 - 创建自己的image

目录 基本概念常用命令使用docker compose启动脚本创建自己的image 使用Docker是现在最为流行的软件发布方式&#xff0c; 本系列将阐述Docker的基本概念&#xff0c;常用命令&#xff0c;启动脚本和如何生产自己的docker image。 在我们发布软件时&#xff0c;往往需要把我…

Visual Studio Installer 点击闪退

Visual Studio Installer 点击闪退问题 1. 问题描述2. 错误类型3. 解决方法4. 结果5. 说明6. 参考 1. 问题描述 重装了系统后&#xff08;系统版本&#xff1a;如下图所示&#xff09;&#xff0c;我从官方网站&#xff08;https://visualstudio.microsoft.com/ ) 下载了安装程…

OpenAI 推出ChatGPT Edu,为高校定制版本

近日&#xff0c;OpenAI 宣布推出 ChatGPT Edu&#xff0c;这是一款专为高校打造的 ChatGPT 版本&#xff0c;旨在帮助学生、教师、研究人员和校园运营部门以负责任的方式部署和使用 AI。 ChatGPT Edu 由 GPT-4o 提供支持&#xff0c;具备强大的文本和图像推理能力&#xff0c;…

FS212E 系列PD协议

PD快充协议芯片FS212EL、FS212EH可以智能的识别插入的手机类型&#xff0c;选择最为合适的协议应对手机快充需要。兼容多类USB Type-C协议&#xff0c;包括TypeC协议、TypeC PD2.0、TypeC PD3.0、TypeC PD3.2等协议。集成OPTO输出&#xff0c;通过电阻直驱反馈光耦。FS212E 的调…

hexo init命令报错:Error: EPERM: operation not permitted, mkdir ‘D:\‘

我用的是git bash通过hexo init安装hexo的&#xff0c;但是报错如下&#xff1a; $ hexo init INFO Cloning hexo-starter https://github.com/hexojs/hexo-starter.git fatal: unable to access https://github.com/hexojs/hexo-starter.git/: HTTP/2 stream 1 was not clos…

Firebase Local Emulator Suite详解

文章目录 Firebase Local Emulator Suite 组件安装和使用步骤1. 安装 Firebase CLI2. 初始化 Firebase 项目3. 配置模拟器4. 启动模拟器5. 配置应用程序使用本地模拟器 常见用途 Firebase Local Emulator Suite 是一组本地服务&#xff0c;可以模拟 Firebase 平台的在线服务&am…

以sqlilabs靶场为例,讲解SQL注入攻击原理【25-31关】

【Less-25】 首先分析源码 发现把 SQL语句中的 or、and 替换成了空格&#xff0c;这就导致无法使用之前的sql注入方式。 解决方案&#xff1a;用 && 代替 and &#xff0c; 用 || 代替 or &#xff0c; 而且&在url中有特殊含义&#xff0c;如果直接使用会有问题&a…

如何在镜像中安装固定版本的node和npm

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、使用 Dockerfile 创建自定义镜像二、如何安装固定版本的node及npm总结 前言 最近在做前端工程化相关的内容&#xff0c;需要在一个镜像内安装固定版本的 N…

redis 高可用及哨兵模式 @by_TWJ

目录 1. 高可用2. redis 哨兵模式3. 图文的方式让我们读懂这几个算法3.1. Raft算法 - 图文3.2. Paxos算法 - 图文3.3. 区别&#xff1a; 1. 高可用 在 Redis 中&#xff0c;实现 高可用 的技术主要包括 持久化、复制、哨兵 和 集群&#xff0c;下面简单说明它们的作用&#xf…

Linux共享内存创建和删除

最近项目中使用到了共享内存记录下 创建共享内存: 删除共享内存: 代码: #include <stdio.h> #include <stdlib.h> #include <string.h> #include <fcntl.h> #include <sys/mman.h> #include <sys/stat.h> #include <u…

计算机视觉与模式识别实验1-3 图像滤波

文章目录 &#x1f9e1;&#x1f9e1;实验流程&#x1f9e1;&#x1f9e1;1. 对图像加入椒盐噪声&#xff0c;并用均值滤波进行过滤2.对图像加入高斯噪声&#xff0c;并用高斯滤波进行过滤3.对图像加入任意噪声&#xff0c;并用中值滤波进行过滤4.读入一张灰度图像&#xff0c;…

【前端开发--css学习笔记】CSS超详细的学习笔记。前端开发css学习笔记(非常详细,适合小白入门)

二&#xff0c;CSS学习笔记 1&#xff0c;CSS语法 1-1 CSS 实例 CSS声明总是以分号 ; 结束&#xff0c;声明总以大括号 {} 括起来: <!DOCTYPE html> <html> <head> <meta charset"utf-8"> <title>菜鸟教程(runoob.com)</title…

非对称密钥:应用场景

public class EncryptionAndSignatureExample { public static void main(String[] args) throws Exception {// 生成公私钥对KeyPairGenerator keyPairGenerator KeyPairGenerator.getInstance("RSA");keyPairGenerator.initialize(1024);KeyPair keyPair keyPai…