Python3学习笔记(二):__repr__和__str__的思考和理解

    最近一下子学了很多的知识点,导致我有点没反应过来,粗略的在草稿纸记了点自己的想法,趁休息的时间将它敲到博客里面去,免得丢失,这一篇写的挺废话的,有点啰嗦,本篇的重点是第二段程序后开始的总结和后面的几个细节问题。

    关于__repr__和__str__这连个东西,我最开始就感觉有点难以理解,百度来的东西看了很多遍,定义都能背下来了,随口一说就是一个给机器看的一个给人看的,但是感觉只理解了最表面,当然不是网上大佬给的答案不够好,而是某些东西还是需要自己去思考,然后有一番自己的理解,然后写点程序验证下,这才是最好的。    

    首先,我把这两个的东西对实例对象的操作称为“渲染”,可能不太准确,但是我觉这么叫方便我去理解。

    第一步先写一个简单的类:MyNumber,先来理解下在输出实例的时候,str和repr的操作方式

   class MyNumber:
        def __init__(self,value):
            self.data = value
        def __str__(self):
            print('正在调用__str__方法,转换为普通字符串')
            s = '自定义数据%d'%self.data
            return s
        def __repr__(self):

             print('正在调用__str__方法,转换为普通字符串')
             s = 'MyNumber(%d)'%self.data

             return s

         n1 = MyNumber(100),

    这个类里面除了init方法还有str和repr,首先,我们在用print输出任何东西的时候,都会有一个渲染步骤,而且默认的就是用str进行渲染,因为任何一样东西都可以看做一个对象,那么它必有一个类型,如果它的类里面没有定义str和repr也没关系,object里面定义了str和repr,object是一切类的父类,所以输出的对象一定会是渲染过的。这个类里面自己写了str和repr,它覆盖了object里面的str和repr,相当于print的重定向。

    接下来就是输出了,print(n1)和print(str(n1))是一样的效果的,因为他们都会调用类里面的str方法,其中print(n1)是默认调用str的。print(repr(n1))的结果就不一样了,它会调用这个类里面的repr方法。

    接下来再弄一段来记下repr的用法和两者的区别。

    class MyInteger:
        def __init__(self,v):
            self.data = v
        def __repr__(self):
            return 'MyInteger(%d)'%self.data
        def __abs__(self): 
        '''此方法用于制定abs(obj)函数取值时返回的结果'''
            if self.data < 0:
                return MyInteger(-self.data) #用-self.data创建一个新的对象返回回去
            return MyInteger(-self.data)
    i1 = MyInteger(-100)
    print(i1)  #等同与print(str(i1))
    n = abs(i1)

    print(n)

    这一段程序比较有意思,先来配合第一段程序来总结下str和repr的调用规则。

    调用 print(i1)  (#等同与print(str(i1)))的时候,解释器第一个寻找的就是i1这个类的方法里面有没有重新定义str,如果没有,那么它第二步会去寻找这个类里面有没有重新定义repr,如果有则会用类方法的repr,如果还没有,那么解释器会找这个类的上一层父类,按同样的规则进行寻找,直到最后找到了object,然后用object的str方法,将该对象的内容转成字符串,最后输出到终端。

    调用print(repr(i1))的时候就不一样了,repr只会调用repr方法,当自定义的类中没有重写repr方法的时候,它会直接找上一级的父类中有没有repr方法,而不会考虑调用str方法。

    总的来说,repr方法比较傲娇,而str方法就比较随意,所以repr的用法就会像这一段程序一样,当我要输出一个需要自己加工的数据的时候,用object的str和repr显然不够,那么就需要在自己的类中重新写一个repr的方法,这样,调用print(XXX)的时候,这个类里面的repr方法就会被调用,这段程序里面,repr调用的意义就输输出了一个段字符串用做提示,这一是一般比较常见的用法。

    最后再来总结一些东西,除了顺序之外还有一些细节。

        1.几乎所有的函数重构会遵循一些返回值规则,str和repr也不例外,自己重构这个函数的时候写得返回值必须是字符串类型,这个规则被写在了解释器的骨子里,试想下,object里定义这两个东西就是为了输出字符串给人或机器看,结果自己重构了一遍返回了个int型的值,解释器也会很苦恼怎么把int的值显示在终端上,干脆就报错了。

        2.所谓给人看和给机器看的意思最直观的就是用eval函数进行测试,eval函数里是需要一个表达式,经过测试就能明白,str返回的是个字符串,而repr返回一个能代表此对象的表达式字符串,这个表达式会被eval翻译,结果就是调用repr时传入的对象,eval(repr(obj))=obj。而str这么做就会报错。

        3.以前经常会有'hello %s'%word  一类的写法,这里%s就是代表了str的类型,其实repr类型对应的是%r,但是都用%s貌似也不会出错,不过还是区分一下,显得更专业一些。

        4.一个小细节,算是比较容易出问题的细节,以第二段程序为例,如果我把print(i1)写成print(i1.data)会怎么样,结果是会直接输出这个实例的属性的值,而且不会调用这个类里面的str和repr方法,因为print里面放的不是一个实例对象,而是该实例的一个属性,所以解释器会直接调用object里面的str,将值转成字符串并输出到了终端,所以一般自己写的类里面重构的repr,一般都是用来自定义的去描述一个实例对象的,如果需要带上实例属性,那就像这一段程序一样,在返回的时候把实例属性插进字符串里面好了。


评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值