python中__str__与__repr__
(1)背景
python中,对于类(自定义类)的实例对象的默认显示既没有太大用处,也不美观。比如:
- class adder:
- def __init__(self,value=0):
- self.data=value #初始化数据
- def __add__(self,other):
- self.data+=other
- 6>>> x=adder()
>>>print(x)
<__main__.adder. object at 0x.....>- >>>x
<__main__.adder object at 0x......>
而通过__str__或者__repr__,可以定制化(costomise)显示,比如,下面代码中,在子类中定义了一个返回实例字符的__repr__方法。注意,在python3中所有对象都继承了object的__str__,也就是实例对象的默认显示。
- >>>class addrepr(adder):
- def __repr__(self):
- return 'addrepr(%s)'% self.data
- >>>x=addrepr(2) #运行__init__
- >>>x+1 #运行__add__
- >>>x #运行__repr__
- addrepr(3)
- >>>print(x) #运行__repr__
- addrepr(3)
- >>>str(x),repr(x) #均运行__repr__
- ('addrepr(3)','addrepr(3)')
当类实例化对象被打印或者转化为字符时,如果定义了__repr__(或者__str__),那么该__repr__(或者__str__)将被自动调用,这里__repr__用了最基本的字符格式来将self.data转化为友好的字符显示。
(2)为什么要用两种显示方法
虽然__str__与__rer__的作用都是为了获得更友好的字符显示,但对于代码的设计有一些细微的区别。
(a)对于print和str内建函数,程序会首先尝试__str__函数,如果没有__str__函数,则尝试__repr__函数,如果没有__repr__函数,则选用默认显示;
(b)在其他情况下,比如交互式回应(interactive echoes),repr函数,和嵌套中,__repr__被调用,一般地,它应该为开发者返回较为详细的显示。
下面通过代码说明两种方法的不同:
- >>>class addstr(adder):
- def __str__(self):
- return '[value:%s]'% self.data
- >>>x=addstr(3)
- >>>x #默认显示
- <__main__.addstr object at 0x....>
- >>>print(x) #调用__str__
- [value:4]
- >>>str(x),repr(x)
- ('[value:4]','<__main__.addstr object at 0x...>
(c)如果同时定义了两种方法,那么可以在不同情况下,支持不同的显示。如下面代码:
- >>>class addboth(adder):
- def __str__(self):
- return '[value:%s]'%self.data
- def __repr__(self):
- return 'addboth(%s)'% self.dat
- >>>x=addboth(4)
- >>>x+1
- >>>x #调用__repr__
- addboth(5)
- >>>print(x) #调用__str__
- [value:5]
- >>>str(x),repr(x) #分别调用__str_,__repr__
- ('[value:5]','addboth(5)')
总结下来以上几点就是:只有在print(),str()时,才会调用__str__()(如果没有__str__则调用__repr__),其他情况均调用__repr__,如交互式情况下单独显示,repr()等。
(3)使用的三点注意
(a)首先是__str__和__repr__必须均返回字符,返回其他类型,将会报错,所以必要的话必须确保它们进行字符转换(比如str,%s)。
(b)根据容器(container)的字符转换,仅有当对象出现在print的顶层时,才会调用__str__;嵌套在大的对象里的对象显示,将仍调用__repr__,下面代码说明了这一点:
- >>>class Printer:
- def __init__(self,value):
- self.value=value
- def __str__(self):
- return str(self.value)
- >>>objs=[Printer(2),Printer(3)]
- >>>for x in objs:print(x)
- 2
- 3
- >>>print(objs)
- [<__main__.Printer object at 0x....>]
- >>>objs
- [<__main__.Printer object at 0x....>,<__main__.Printer object at 0x....>]
为确保不论有无容器,在所有情况下显示设定的显示模式,用__repr__,不用__str__,用如下代码进行说明:
- >>> class Printer:
- def __init__(self,value):
- self.val=value
- def __repr__(self): #如果没有__str__,调用__repr__
- return '%s'% self.val
- >>> objs=[Printer(2),Printer(3)]
- >>> for x in objs:print(x)
- 2
- 3
- >>> print(objs) #调用__repr__
- [2, 3]
- >>> objs
- [2, 3]
(c)第三,也是最为微妙的,显示方法在极少情况下有时又也有可能触发无限迭代循环(infinite recursion loops),因为一些对象的显示包括了其他对象的的显示,而一个显示触发了正在被显示的对象的显示,因而进入无限循环中。如下代码:
- """
- this scripts is intended to illustrate the infinite recursion loops
- caused by __repr__ overloading methods. displaying the value of a method,line10 in this script, can trigger the __repr__
- of the class method, then the __repr__ method is called again, and the infinite recursion loops happen.
- """
- class Base:
- def __init__(self):
- self.data=1
- def print0(self):
- pass
- def print1(self):
- a=str(getattr(self,'print0')) #Caution! getattr(object,attrname),attrname shall be string.
- return a
- class Normal(Base):
- def __str__(self):
- return '%s'% self.print1()
- class Recursion(Base):
- def __repr__(self):
- return '%s'% self.print1()
- if __name__=='__main__':
- a=Normal()
- b=Recursion()
- print(a)
- try:
- print(b)
- except RecursionError:
- print('A recusion error happens')
运行结果为:
- <bound method Base.print0 of <__main__.Normal object at 0x02E68450>>
- A recusion error happens
python中__str__与__repr__的更多相关文章
- Python中__str__和__repr__的区别
Python有一个内置的函数叫repr,它能把一个对象用字符串的形式表达出来以便辨认,这就是“字符串表示形式”.repr就是通过__repr__这个特殊方法来得到一个对象的字符串表示形式.如果没有实现 ...
- 浅谈python中__str__和__repr__的区别
很多时候我们在创建一个类的时候,在终端打印类或者查看的时候一般都不会得到一个太满意的结果 class T: def __init__(self): self.color="red" ...
- python中__str__与__repr__的区别
__str__和repr __str__和__repr__都是python的内置方法,都用与将对象的属性转化成人类容易识别的信息,他们有什么区别呢 来看一段代码 from math import hy ...
- python的__str__()和__repr__()方法
__str__()和__repe__()方法定义一个值通过print打印时返回时的显示样式 l=list('hello') print(l) #['h', 'e', 'l', 'l', 'o'] cl ...
- python特殊函数__str__、__repr__和__len__
1.__str__ 首先介绍__str__ class Students(object): def __init__(self, *args): self.names = args # def __s ...
- python中的类与继承
Class 类的定义以及实例的建立 Python中,类通过 class 关键字定义. 例如最简单的一个类定义可以为: class Person(object): pass Python 的编程习惯,类 ...
- python中定制类
1.python中__str__和repr 如果要把一个类的实例变成 str,就需要实现特殊方法__str__(): class Person(object): def __init__(self, ...
- 面向对象相关概念与在python中的面向对象知识(魔法方法+反射+元类+鸭子类型)
面向对象知识 封装 封装的原理是,其成员变量代表对象的属性,方法代表这个对象的动作真正的封装是,经过深入的思考,做出良好的抽象(设计属性时用到),给出“完整且最小”的接口,并使得内部细节可以对外透明( ...
- Python中__repr__和__str__区别
Python中__repr__和__str__区别 看下面的例子就明白了 class Test(object): def __init__(self, value='hello, world!'): ...
随机推荐
- 加拿大Assignment写作如何靠第一句话来吸睛?
最近有留学加拿大的同学咨询我们如何写好assignment的首句,关于话题背景的引入,如何才能自然精妙,让老师对后文充满期待.小编用一句话总结今天的策略:陈述一个跟话题直接相关的事实.“直接相关”,保 ...
- 基于RabbitMQ的MQTT协议及应用
MQTT的开源代码地址先贴在这里:https://github.com/mqtt/mqtt.github.io/wiki/servers MQTT定义: MQTT(Message Queuing Te ...
- SP2-0157: unable to CONNECT to ORACLE after 3 attempts, exiting SQL*Plus
[oracle@jtwy02 ~]$ sqlplus '/as sysdba' SQL*Plus: Release 11.2.0.4.0 Production on Sat Oct 13 14:14: ...
- 2019 OI日记
// 我觉得记日记是个好习惯吧 毕竟指不定哪天就学不下去了 就AFO了 就没有梦了 // [置顶]活跃于你谷普及训练场.ybt(没底气说全部).loj(提高基础部分) //优先级从前往后 因为 ...
- vue 中使用 echarts 自适应问题
echarts 自带的自适应方法 resize() 具体用法: let xxEcharts = this.$echarts.init(document.getElementById('xxx')) ...
- 清北学堂例题 LUOGU2519 【HAOI2011】PROBLEM A
题目描述 一次考试共有n个人参加,第i个人说:“有ai个人分数比我高,bi个人分数比我低.”问最少有几个人没有说真话(可能有相同的分数) 输入格式 第一行一个整数n,接下来n行每行两个整数,第i+1行 ...
- C++代做,C++编程代做,C++程序代做,留学生C++ Lab代写
C++代做,C++编程代做,C++程序代做 我们主要面向留学生,广泛接美加澳国内港台等地编程作业代写,中英文均可. C语言代写 C++代写 Python代写 Golang代写 Java代写 一年半的时 ...
- 003.CI4框架CodeIgniter, 控制器Controllers的访问地址
01.我们新建一个System文件夹,然后创建一个Login.php类,代码如下: <?php namespace App\Controllers\System; use App\Control ...
- php中date('Y/m/d',time())显示不对
一. 时间不对是因为没设置时区 在xampp/php/php.ini中ctrl + f 查找date.timezone 该行默认注释,去掉 ; 修改为 date.timezone = PRC 二 上述 ...
- Django(二) 模板:基本使用、模板语法、模板继承
一.模板的使用实战 接:https://blog.csdn.net/u010132177/article/details/103788677 参考:https://docs.djangoproject ...