python中魔术方法__str__与__repr__的区别

news2025/1/8 5:55:59

在这里插入图片描述

在Python中,__str____repr__是两个常见的魔法方法(也称为双下方法或dunder方法),它们用于定义对象的字符串表示形式。它们的主要区别在于它们的用途和使用场景。

__str__

  • 用途__str__方法用于为用户提供一个易读的字符串表示形式。当使用str()函数或print()函数打印对象时,会调用该方法。

  • 目标用户:面向最终用户,旨在提供一个易读且有意义的描述。

  • 示例

    class Person:
        def __init__(self, name, age):
            self.name = name
            self.age = age
    
        def __str__(self):
            return f"Person(name={self.name}, age={self.age})"
    
    p = Person("Alice", 30)
    print(p)  # 输出:Person(name=Alice, age=30)
    

__repr__

  • 用途__repr__方法用于为开发者提供一个更正式的字符串表示形式,通常应尽可能地提供一个准确且完整的描述。当使用repr()函数或在交互式解释器中直接输入对象时,会调用该方法。

  • 目标用户:面向开发者,旨在提供一个详细且准确的描述,通常应该包含足够的信息以便于通过该字符串重新创建对象。

  • 示例

    class Person:
        def __init__(self, name, age):
            self.name = name
            self.age = age
    
        def __repr__(self):
            return f"Person(name={self.name!r}, age={self.age!r})"
    
    p = Person("Alice", 30)
    print(repr(p))  # 输出:Person(name='Alice', age=30)
    

综合示例

如果一个类同时定义了__str____repr__,它们会分别在不同的上下文中被调用:

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def __str__(self):
        return f"Person(name={self.name}, age={self.age})"

    def __repr__(self):
        return f"Person(name={self.name!r}, age={self.age!r})"

p = Person("Alice", 30)

# 使用 str() 或 print() 函数时调用 __str__
print(str(p))  # 输出:Person(name=Alice, age=30)
print(p)       # 输出:Person(name=Alice, age=30)

# 使用 repr() 或在交互式解释器中直接输入对象时调用 __repr__
print(repr(p)) # 输出:Person(name='Alice', age=30)

总结

  • __str__:提供面向用户的友好字符串表示。
  • __repr__:提供面向开发者的正式且完整的字符串表示,通常应该包括可以重新创建对象的信息。

_repr__方法的设计理念

在Python中,__repr__方法的设计理念之一是提供一个准确且完整的字符串表示形式,使得该字符串表示形式尽可能地能够重新创建该对象。这意味着理想情况下,使用eval()函数可以通过该字符串表示形式重新生成一个相同的对象实例。

为什么需要这种设计?

  1. 调试和开发:在调试和开发过程中,开发者希望通过查看对象的__repr__输出来获取尽可能多的信息,以便理解对象的当前状态。
  2. 日志记录:在记录日志时,使用__repr__可以确保记录的信息详尽且准确,有助于在日后分析和调试。
  3. 对象重建:虽然在实际应用中不总是使用eval()来重建对象,但这种设计理念保证了__repr__输出尽可能地完整和精确。

示例说明

以下是一个类及其__repr__实现的示例,该实现能够提供重新创建对象的信息:

class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    def __repr__(self):
        return f"Person(name={self.name!r}, age={self.age!r})"

p = Person("Alice", 30)
print(repr(p))  # 输出:Person(name='Alice', age=30)

# 使用 eval() 来重新创建对象
p_copy = eval(repr(p))
print(p_copy)   # 输出:Person(name='Alice', age=30)
print(p == p_copy)  # 输出:False (因为它们是两个不同的对象实例)
print(p.name == p_copy.name and p.age == p_copy.age)  # 输出:True (因为它们的属性值相同)
  • __repr__方法返回了一个字符串,这个字符串看起来像一个调用类构造函数的表达式,并且包含了创建该对象所需的所有信息。
  • 通过eval(repr(p))可以重新创建一个具有相同属性的新对象p_copy

注意事项

  • 安全性:在实际开发中,应谨慎使用eval()函数,因为它会执行传入的字符串,这可能带来安全风险。如果字符串包含恶意代码,可能会导致不可预期的后果。
  • 复杂对象:对于一些复杂对象(如涉及外部资源、网络连接、文件句柄等),__repr__可能无法完全捕捉所有信息。在这种情况下,__repr__应尽量提供尽可能多的有用信息,同时注意避免安全和隐私问题。

总结

__repr__方法旨在提供一个正式且详细的对象表示形式,使开发者能够通过该表示形式了解对象的内部状态,并在理想情况下能够重新创建该对象。这种设计理念在调试、日志记录和对象序列化等场景中尤为有用。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

适合营销的叙事可视化

背景 数据可视化与数据故事化的差异和相似点,以及它们如何协同工作,将你的数据转化为清晰、简洁、可操作的信息,以便您的组织使用。 什么是数据可视化? 数据可视化通过图像传达信息——这是你所收集数据的视觉表示。通过提供原…

可以自定义的文字识别OCR

可以自定义的文字识别OCR 什么是OCR文档自学习自定义模板单证票据信息抽取操作体验 这里提到的可以自定义的文字识别OCR ,其实就是OCR文档自学习。 什么是OCR文档自学习 什么是OCR文档自学习呢?OCR文档自学习,是面向“无算法基础”的企业与个…

【外汇天眼】交易智慧:遵循趋势,稳中求胜

在交易中,新手往往因对未来走势的不确定性感到恐惧,从而不断要求对市场进行全面分析。这种需求反映了他们在投机心理幼稚期缺乏安全感的表现。有些勤奋的交易者甚至在做单前、持仓时和寻找出局理由时都在不断分析行情。然而,这种过度分析真的…

经典的带环链表问题(链表补充)

环形链表1 运用快慢指针的方法,fast ,slow从头节点出发,快指针走两步,慢指针走一步,若有环,快指针先进环,后续如果慢指针和快指针相遇,则链表带环。转换成了追击问题。 struct ListNode {int v…

第二证券股市资讯:半导体,突发!

半导体又现突发! 商场忽然传出,拜登政府正在考虑约束我国获取应用在人工智能(AI) 芯片上的全栅级晶体管技能(Gate-all-around, GAA) ,但不过现在还不清楚美国官员何时会做出最终决议。从趋势来看,这意味着…

高温预警,快收下这份机房运维攻略

高温预警 华东区即将迎来最强高温,根据历史经验,数据机房在夏季高温环境导致设备温度过高,宕机事件明显增加,为保障系统健康稳定运行,需要针对数据机房空调、设备的运行状态及环境进行检查,并同时期开展防尘…

八股文系列Spark

为什么Spark 比 MapReduce 更快 DAG相比hadoop的mapreduce在大多数情况下可以减少磁盘I/O次数 因为mapreduce计算模型只能包含一个map和一个reduce,所以reduce完后必须进行落盘,而DAG可以连续shuffle的,也就是说一个DAG可以完成好几个mapreduce&#xf…

vue 应用测试(一) --- 介绍

vue 应用测试(一) ---介绍 前端测试简介组件测试Jest 测试框架简介其他测试框架 第一个测试避免误报如何组织测试代码 组件挂载Vue2 组件挂载的方式Vue3 的挂载方式vue-test-utils挂载选项 如何调试测试用例参考小结 前端测试简介 软件测试:…

康谋分享 | 自动驾驶联合仿真——功能模型接口FMI(一)

功能模型接口FMI(Functional Mock-up Interface)是一个开放且与工具解耦的标准。FMI包含了一个C-API(接口),一个用于描述接口的XML文件以及可交换的功能模型单元FMU(Functional Mock-up Unit)&a…

通过nerdctl+buildctl编译发布go程序docker镜像

1 nerdctl安装 下载: wget -c https://github.com/containerd/nerdctl/releases/download/v1.7.6/nerdctl-full-1.7.6-linux-amd64.tar.gz 解压: tar -zxf nerdctl-full-1.7.6-linux-amd64.tar.gz -C /usr/local/nerdctl 配置: /usr/local/nerdctl/…

6月报名 | 海克斯康Actran风机类气动噪声分析培训

您好!感谢您长期以来对优飞迪科技与海克斯康的关注与支持。我们诚邀您参加海克斯康Actran风机类气动噪声分析培训,特邀海克斯康原厂讲师将以实操为基础,结合真实案例,手把手帮您解锁噪声仿真关键技术。 活动主题: 海…

【架构之路】微服务中常用的几种通信方式

2024年,计算机相关专业还值得选择吗? 强烈推荐 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站:人工智能 引言 微服务架构由于其灵活性、高可扩展性和易维护性&am…

mysql打开远程访问

这里写目录标题 1.使用navicat进入命令控制板,进入use mysql;2.查询用户表3.更新user表中root用户域属性,%表示允许外部访问4.执行以上语句之后再执行,FLUSH PRIVILEGES;5. 执行授权语句 1.使用navicat进入命令控制板,进入use mysql; use mysql;2.查询用…

NOSQL -- MOGODB

Mogodb简介: 是一个开源的, 高性能, 无模式的文档型数据库. NoSql数据库产品当中的一种, 也是最像关系型数据库的非关系型数据库 使用场景: 针对不同的应用场景, 以及其对应的修改对应数据的频率, 我们可以以此选择需要哪一种类型的数据库 Mongo的使用: 启动: 在解压完成之后…

动手学操作系统(六、获取物理内存容量)

动手学操作系统(六、获取物理内存容量) 在上一节中,我们介绍了保护模式和实模式的区别,保护模式的最大特点是“大”,“大”是指寻址空间大,在进入保护模式之后,我们还将要接触虚拟内存、内存管…

基于pytorch实现的DenseUnet医学图像分割(腹部多脏器)

1、前言 本章将介绍将densenet的主干网络引入unet中 官方实现的代码:kits19-challenge/network at master nitsaick/kits19-challenge (github.com) 本章实现的项目目录如下: 主要代码有train、evaluate、predict脚本 2、代码介绍 数据预处理脚本 数据…

全能型施耐德可编程控制器M241介绍

施耐德M241是一款通信强大、定位控制、丰富扩展于一身的全能型可编程控制器,适用于具有速度控制和位置控制功能的高性能一体型设备。其内置以太网通信端口,可以提供FTP和网络服务器功能,能够更为便捷地整合到控制系统架构中,通过智…

【0基础学爬虫】爬虫基础之自动化工具 DrissionPage 的使用

概述 前三期文章中已经介绍到了 Selenium 与 Playwright 、Pyppeteer 的使用方法,它们的功能都非常强大。而本期要讲的 DrissionPage 更为独特,强大,而且使用更为方便,目前检测少,强烈推荐!!&a…

Spark Streaming 概述及入门案例

一、介绍 1. 不同的数据处理 从数据处理的方式: 流式数据处理(Streaming)批量数据处理(Batch) 从数据处理的延迟: 实时数据处理(毫秒级别)离线数据处理(小时或天级别) 2. 简介 SparkStreaming 是一个准实时(秒或分钟级别)、微批量的数据处理框架Spa…

了解侧信道攻击基础知识

人们通常认为特洛伊木马、恶意软件和其他形式的黑客攻击等漏洞是的威胁;然而,从 EE 的角度来看,安全性具有全新的含义。  事实上,许多的安全威胁都是基于硬件的,攻击者可以直接从运行我们的安全加密软件的硬件中窃取…