课时44:魔法方法:简单定制
目录:
一、简单定制
二、课时44课后习题及答案
****************
一、简单定制
****************
基本要求:
1>> 定制一个计时器的类
2>> start和stop方法代表启动计时和停止计时
3>> 假设计时器对象t1,print(t1)和直接调用t1均显示结果
4>> 当计时器未启动或已经停止计时,调用stop方法会给予温馨的提示
5>> 两个计时器对象可以进行相加:t1 + t2
6>> 只能使用提供的有限资源完成
你需要这些资源:
1>> 使用time模块的localtime方法获取时间
【扩展阅读】:time 模块详解(时间获取和转换)
2>> time.localtime返回struct_time的时间格式
3>> 表现你的类:__str__ ()和 __repr__()魔法方法
>>> class A: def __str__(self): return "小甲鱼是帅哥" >>> a = A() >>> print(a) 小甲鱼是帅哥 >>> a <__main__.A object at 0x0000020BB2E537F0> >>> class B: def __repr__(self): return "小甲鱼是帅哥" >>> b = B() >>> b 小甲鱼是帅哥
有了这些知识,可以开始编写代码了:
import time as t class MyTimer: #开始计时 def start(self): self.start = t.localtime() print("计时开始...") #停止计时 def stop(self): self.stop = t.localtime() print("计时结束!") """ 好,万丈高楼平地起,把地基写好后,应该考虑怎么进行计算了。 localtime()返回的是一个时间元组的结构,只需要在前边6个元素, 然后将stop的元素依此减去start对应的元素,将差值存放在一个新的列表里: """ #停止计时 def stop(self): self.stop = t.localtime() self._calc() print("计时结束!") # 内部方法,计算运行时间 def _calc(self): self.lasted = [] self.prompt = "总共运行了" for index in range(6): self.lasted.append(self.stop[index] - self.start[index]) self.prompt += str(self.lasted[index]) print(self.prompt)
>>> t1 = MyTimer() >>> t1.start() 计时开始... >>> t1.stop() 总共运行了000008 计时结束!
已经基本实现计时功能了,接下来需要完成“print(t1)和直接调用t1均显示结果”,那就要通过重写__str__()和__repr__()魔法方法来实现:
def __str__(self): return self.prompt __repr__ = __str__
>>> t1 = MyTimer() >>> t1.start() 计时开始... >>> t1.stop() 总共运行了000004 计时结束! >>> t1 总共运行了000004
似乎做得很不错了,但这里还有一些问题。假使用户不按常理出牌,问题就会很多:
>>> t1 = MyTimer() >>> t1 Traceback (most recent call last): File "<pyshell#10>", line 1, in <module> t1 File "C:\Users\14158\AppData\Local\Programs\Python\Python37\lib\idlelib\rpc.py", line 617, in displayhook text = repr(value) File "C:\Users\14158\Desktop\lalallalalal.py", line 36, in __str__ return self.prompt AttributeError: 'MyTimer' object has no attribute 'prompt'
当直接执行t1的时候,Python会调用__str__()魔法方法,但它却说这个类没有prompt属性。prompt属性在哪里定义的?在_calc()方法里定义的,对不?但是没有执行stop()方法,_calc()方法就没有被调用到,所以也就没有prompt属性的定义了。
要解决这个问题也很简单,大家应该还记得在类里边,用得最多的一个魔法方法是什么?是__init__()嘛,所有属于实例对象的变量只要在这里边先定义,就不会出现这样的问题了。
def __init__(self): self.prompt = "未开始计时!" self.lasted = [] self.start = 0 self.stop = 0
>>> t1 = MyTimer() >>> t1 未开始计时! >>> t1.start() Traceback (most recent call last): File "<pyshell#16>", line 1, in <module> t1.start() TypeError: 'int' object is not callable
这里又报错了(当然是故意的),先检查一下出现了什么问题?
Python这里抛出了一个异常:TypeError: 'int' object is not callable
仔细瞧,在调用start()方法的时候报错,也就是说,Python认为start是一个整型变量,而不是一个方法。为什么呢?大家看__init__()方法里,是不是也命名了一个叫做self.start的变量,如果类中的方法名和属性同名,属性会覆盖方法。
好了,让我们把self.start和self.stop都改为self.begin和self.end吧!
现在程序没什么问题了,但显示的时间不怎么好看,希望按章年月日时分秒来显示,所以这里添加一个列表用于存放对应的单位,然后再适当的地方增加温馨提示:
import time as t class MyTimer: def __init__(self): self.unit = ['年', '月', '天', '小时', '分钟', '秒'] self.prompt = "未开始计时!" self.lasted = [] self.begin = 0 self.end = 0 def __str__(self): return self.prompt __repr__ = __str__ def __add__(self, other): prompt = "总共运行了" result = [] for index in range(6): result.append(self.lasted[index] + other.lasted[index]) if result[index]: prompt += (str(result[index]) + self.unit[index]) return prompt # 开始计时 def start(self): self.begin = t.localtime() self.prompt = "提示:请先调用 stop() 停止计时!" print("计时开始...") # 停止计时 def stop(self): if not self.begin: print("提示:请先调用 start() 进行计时!") else: self.end = t.localtime() self._calc() print("计时结束!") # 内部方法,计算运行时间 def _calc(self): self.lasted = [] self.prompt = "总共运行了" for index in range(6): self.lasted.append(self.end[index] - self.begin[index]) if self.lasted[index]: self.prompt += (str(self.lasted[index]) + self.unit[index]) # 为下一轮计时初始化变量 self.begin = 0 self.end = 0
最后再写一个魔法方法__add__(),让两个计时器对象相加会自动返回时间的和:
def __add__(self,other): prompt = "总共运行了" result = [] for index in range(6): result.append(self.lasted[index] + other.lasted[index]) if result[index]: prompt += (str(result[index]) + self.unit[index]) return prompt
>>> t1 = MyTimer() >>> t1 未开始计时! >>> t1.stop() 提示:请先调用 start() 进行计时! >>> t1.start() 计时开始... >>> t1 提示:请先调用 stop() 停止计时! >>> t1.stop() 计时结束! >>> t1 总共运行了7秒 >>> t2 = MyTimer() >>> t2.start() 计时开始... >>> t2.stop() 计时结束! >>> t1 + t2 '总共运行了18秒'
看上去代码不错,也能正常计算了。但是这个程序还有几点不足的地方:
(1)如果开始计时的时间是(2022年2月22日16:30:30),停止时间是(2025年1月23日15:30:30),那按照我们用停止时间减开始时间的计算方式就会出现负数(3年-1月1天-1小时),你应该对此做一些转换。
(2)现在的计算机速度都非常快,而我们这个程序最小的计算单位却只是秒,精度是远远不够的。
*******************************
二、课时44课后习题及答案
*******************************