最近一下子学了很多的知识点,导致我有点没反应过来,粗略的在草稿纸记了点自己的想法,趁休息的时间将它敲到博客里面去,免得丢失,这一篇写的挺废话的,有点啰嗦,本篇的重点是第二段程序后开始的总结和后面的几个细节问题。
关于__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,一般都是用来自定义的去描述一个实例对象的,如果需要带上实例属性,那就像这一段程序一样,在返回的时候把实例属性插进字符串里面好了。