(1)背景

python中,对于类(自定义类)的实例对象的默认显示既没有太大用处,也不美观。比如:

  1. class adder:
  2. def __init__(self,value=0):
  3. self.data=value #初始化数据
  4. def __add__(self,other):
  5. self.data+=other
  6. 6>>> x=adder()
    >>>print(x)
       <__main__.adder. object at 0x.....>
  7. >>>x
        <__main__.adder object at 0x......>

而通过__str__或者__repr__,可以定制化(costomise)显示,比如,下面代码中,在子类中定义了一个返回实例字符的__repr__方法。注意,在python3中所有对象都继承了object的__str__,也就是实例对象的默认显示。

  1. >>>class addrepr(adder):
  2. def __repr__(self):
  3. return 'addrepr(%s)'% self.data
  4. >>>x=addrepr(2) #运行__init__
  5. >>>x+1 #运行__add__
  6. >>>x #运行__repr__
  7. addrepr(3)
  8. >>>print(x) #运行__repr__
  9. addrepr(3)
  10. >>>str(x),repr(x) #均运行__repr__
  11. ('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__被调用,一般地,它应该为开发者返回较为详细的显示。

下面通过代码说明两种方法的不同:

  1. >>>class addstr(adder):
  2. def __str__(self):
  3. return '[value:%s]'% self.data
  4. >>>x=addstr(3)
  5. >>>x #默认显示
  6. <__main__.addstr object at 0x....>
  7. >>>print(x) #调用__str__
  8. [value:4]
  9. >>>str(x),repr(x)
  10. ('[value:4]','<__main__.addstr object at 0x...>

(c)如果同时定义了两种方法,那么可以在不同情况下,支持不同的显示。如下面代码:

  1. >>>class addboth(adder):
  2. def __str__(self):
  3. return '[value:%s]'%self.data
  4. def __repr__(self):
  5. return 'addboth(%s)'% self.dat
  6. >>>x=addboth(4)
  7. >>>x+1
  8. >>>x #调用__repr__
  9. addboth(5)
  10. >>>print(x) #调用__str__
  11. [value:5]
  12. >>>str(x),repr(x) #分别调用__str_,__repr__
  13. ('[value:5]','addboth(5)')

总结下来以上几点就是:只有在print(),str()时,才会调用__str__()(如果没有__str__则调用__repr__),其他情况均调用__repr__,如交互式情况下单独显示,repr()等。

(3)使用的三点注意

(a)首先是__str__和__repr__必须均返回字符,返回其他类型,将会报错,所以必要的话必须确保它们进行字符转换(比如str,%s)。

(b)根据容器(container)的字符转换,仅有当对象出现在print的顶层时,才会调用__str__;嵌套在大的对象里的对象显示,将仍调用__repr__,下面代码说明了这一点:

  1. >>>class Printer:
  2. def __init__(self,value):
  3. self.value=value
  4. def __str__(self):
  5. return str(self.value)
  6. >>>objs=[Printer(2),Printer(3)]
  7. >>>for x in objs:print(x)
  8.  
  9. 2
  10. 3
  11. >>>print(objs)
  12. [<__main__.Printer object at 0x....>]
  13. >>>objs
  14. [<__main__.Printer object at 0x....>,<__main__.Printer object at 0x....>]

为确保不论有无容器,在所有情况下显示设定的显示模式,用__repr__,不用__str__,用如下代码进行说明:

  1. >>> class Printer:
  2. def __init__(self,value):
  3. self.val=value
  4. def __repr__(self): #如果没有__str__,调用__repr__
  5. return '%s'% self.val
  6.  
  7. >>> objs=[Printer(2),Printer(3)]
  8. >>> for x in objs:print(x)
  9.  
  10. 2
  11. 3
  12. >>> print(objs) #调用__repr__
  13. [2, 3]
  14. >>> objs
  15. [2, 3]

(c)第三,也是最为微妙的,显示方法在极少情况下有时又也有可能触发无限迭代循环(infinite recursion loops),因为一些对象的显示包括了其他对象的的显示,而一个显示触发了正在被显示的对象的显示,因而进入无限循环中。如下代码:

  1. """
  2. this scripts is intended to illustrate the infinite recursion loops
  3. caused by __repr__ overloading methods. displaying the value of a method,line10 in this script, can trigger the __repr__
  4. of the class method, then the __repr__ method is called again, and the infinite recursion loops happen.
  5. """
  6. class Base:
  7. def __init__(self):
  8. self.data=1
  9. def print0(self):
  10. pass
  11. def print1(self):
  12. a=str(getattr(self,'print0')) #Caution! getattr(object,attrname),attrname shall be string.
  13. return a
  14. class Normal(Base):
  15. def __str__(self):
  16. return '%s'% self.print1()
  17. class Recursion(Base):
  18. def __repr__(self):
  19. return '%s'% self.print1()
  20. if __name__=='__main__':
  21. a=Normal()
  22. b=Recursion()
  23. print(a)
  24. try:
  25. print(b)
  26. except RecursionError:
  27. print('A recusion error happens')

运行结果为:

  1. <bound method Base.print0 of <__main__.Normal object at 0x02E68450>>
  2. A recusion error happens

python中__str__与__repr__的更多相关文章

  1. Python中__str__和__repr__的区别

    Python有一个内置的函数叫repr,它能把一个对象用字符串的形式表达出来以便辨认,这就是“字符串表示形式”.repr就是通过__repr__这个特殊方法来得到一个对象的字符串表示形式.如果没有实现 ...

  2. 浅谈python中__str__和__repr__的区别

    很多时候我们在创建一个类的时候,在终端打印类或者查看的时候一般都不会得到一个太满意的结果 class T: def __init__(self): self.color="red" ...

  3. python中__str__与__repr__的区别

    __str__和repr __str__和__repr__都是python的内置方法,都用与将对象的属性转化成人类容易识别的信息,他们有什么区别呢 来看一段代码 from math import hy ...

  4. python的__str__()和__repr__()方法

    __str__()和__repe__()方法定义一个值通过print打印时返回时的显示样式 l=list('hello') print(l) #['h', 'e', 'l', 'l', 'o'] cl ...

  5. python特殊函数__str__、__repr__和__len__

    1.__str__ 首先介绍__str__ class Students(object): def __init__(self, *args): self.names = args # def __s ...

  6. python中的类与继承

    Class 类的定义以及实例的建立 Python中,类通过 class 关键字定义. 例如最简单的一个类定义可以为: class Person(object): pass Python 的编程习惯,类 ...

  7. python中定制类

    1.python中__str__和repr 如果要把一个类的实例变成 str,就需要实现特殊方法__str__(): class Person(object): def __init__(self, ...

  8. 面向对象相关概念与在python中的面向对象知识(魔法方法+反射+元类+鸭子类型)

    面向对象知识 封装 封装的原理是,其成员变量代表对象的属性,方法代表这个对象的动作真正的封装是,经过深入的思考,做出良好的抽象(设计属性时用到),给出“完整且最小”的接口,并使得内部细节可以对外透明( ...

  9. Python中__repr__和__str__区别

    Python中__repr__和__str__区别 看下面的例子就明白了 class Test(object): def __init__(self, value='hello, world!'): ...

随机推荐

  1. git/github使用详解

    介绍:gitHub是一个面向开源及私有软件项目的托管平台,因为只支持git 作为唯一的版本库格式进行托管,故名gitHub. 2018年6月4日,微软宣布,通过75亿美元的股票交易收购代码托管平台Gi ...

  2. shell命令、调度工具、后台执行线程和软连接

    一.shell命令 1.后缀.sh 第一行需要加#!/bin/bash 没有的话,需呀sh 命令执行 示例test.sh: #!/bin/bash date ./test.sh 提示没有权限,此时,需 ...

  3. 01.DesignParttern设计模式,简单工厂,工厂方法,抽象工厂三大工厂的区别与联系

                工厂用来生产对象,对象具有方法和属性. 简单工厂的缺点(简单工厂并不是23中设计模式): 工厂类的职责相对过重,增加新的产品,需要修改工厂类的判断逻辑,违背开闭原则: JDK源 ...

  4. 图片索引 lire

    1:定义  LIRE( Lucene Image Retrieval)相似图像索引和搜索机制 2:资料来源     LIRE官网:http://www.semanticmetadata.net/lir ...

  5. day05-Python运维开发基础(双层循环、pass/break/continue、for循环)

    # ### 双层循环练习 # 十行十列小星星 j = 0 while j<10: # 逻辑代码写在下面 # 打印一行十个小星星 i = 0 while i<10: print(" ...

  6. 需要多个参数输入时-----------------考虑使用变种的Builder模式

    业务需求: 创建一个不可变的Person对象,这个Person可以拥有以下几个属性:名字.性别.年龄.职业.车.鞋子.衣服.钱.房子. 要求: 其中名字和性别是必填项,而其他选填项可以根据情况自由输入 ...

  7. Vue(九)---自定义指令(directive )

    1.无参数 自定义指令的方式:1. 使用Vue.directive 来自定义2. 第一个参数就是 指令名称 xart3. el 表示当前的html dom对象4. 在方法体内就可以通过 innerHT ...

  8. 7.4 Varnish VCL的子程序

  9. c#实现"扫描检测硬件改动"

    public static class Win32Api { public const int CM_LOCATE_DEVNODE_NORMAL = 0x00000000; public const ...

  10. Linux下的文件目录树结构

    Linux下的文件目录及文件结构 一.文件和文件夹 在Linux系统下,一切皆是文件.就连Linux本身也是基于文件表示的操作系统. 1.文件:文件在Linux系统之下,一般分为两种:一是一般性文件, ...