python自定義時鐘類、定時任務(wù)類
這是我使用python寫的第一個類(也算是學(xué)習(xí)面向?qū)ο笳Z言以來正式寫的第一個解耦的類),記錄下改進(jìn)的過程。
分析需求
最初,因?yàn)槭褂胻ime模塊顯示日期時,每次都要設(shè)置時間字符串的格式,挺麻煩,但還是忍了。
后來,在處理多線程任務(wù)時需要實(shí)現(xiàn)定時控制的功能,更麻煩,終于決定自己做一個解決這些問題的通用代碼(雖然網(wǎng)上有現(xiàn)成的模塊,但親手編寫這部分代碼正好能鍛煉一下我的面向?qū)ο缶幊蹋?/p>
分析框架
剛開始,我計劃做一個模仿時鐘的抽象類,讓它獨(dú)立運(yùn)行在一個線程中,讓它提供顯示日期、計時、設(shè)置定時任務(wù)的方法……然而由于缺乏規(guī)劃,編程亂糟糟的,這些方法的代碼和變量交雜在一起,難以入目,更難以擴(kuò)展……氣得重構(gòu)代碼,這次把顯示日期、計時、設(shè)置定時任務(wù)三大功能分別抽象成三個類,相互解耦,各自獨(dú)立運(yùn)行,代碼變得簡潔多了。
ok,舊代碼就藏在git的歷史記錄里吧,這里貼出重構(gòu)后的代碼。
顯示時間的類
import time import threading class _Clock: """ 自定義的時鐘類,用于獲取幾種不同格式的當(dāng)前時間。 decimal : 設(shè)置time_float的精度,控制其保留幾位小數(shù)。 time_diff : 設(shè)置該時鐘與UTC+0時區(qū)的時差。如果不設(shè)置,會自動采用 本地時區(qū)。 """ def __init__(self, name=None, decimal=3, time_diff=None): self.name = name self.decimal = decimal self.time_diff = time_diff self.time_format = "%Y/%m/%d %H:%M:%S" # 時間字符串的格式 @property def time_float(self): """ UTC+0時區(qū)的時間戳,精度由self.decimal決定 """ return round(time.time(), self.decimal) @property def time_int(self): """ UTC+0時區(qū)的時間戳,精度為秒 """ return int(time.time()) @property def time_tuple(self): """ 本地時區(qū)的時間元組 """ if self.time_diff == None: return time.localtime(self.time_int) else: return time.gmtime(self.time_int+self.time_diff) @property def time_str(self): """ 本地時間的格式化字符串 """ return time.strftime(self.time_format, self.time_tuple)
秒表計時的類
class Timer(_Clock):
"""
自定義的計時器,像秒表一樣,可以隨時查看當(dāng)前計時、暫停計時、繼續(xù)計時。
· 創(chuàng)建一個計時器之后,它就會開始計時。
· 默認(rèn)使用time.time()獲取時間,精度為毫秒。
· 可以直接調(diào)用_Clock類的方法來獲取當(dāng)前時間。
"""
def __init__(self, *args, **kwargs):
_Clock.__init__(self, *args, **kwargs)
self.record = [] # 記錄每次使用的 (開始時刻,暫停時刻,計時時長)
self.status = "initial"
self.go()
@property
def count(self):
""" 當(dāng)前計時值 """
count = 0
for line in self.record:
if line[2] == None:
count += self.time_float - line[0]
else:
count += line[2]
return round(count, self.decimal)
def go(self):
""" 開始計時 """
if self.status != "timing":
self.record.append((self.time_float, None, None))
self.status = "timing"
def pause(self):
""" 暫停計時 """
# 如果該計時器在計時中,就暫停它,并計算這一段的計時時長
if self.status == "timing":
last_line = self.record[-1]
self.record.remove(last_line)
current_time = self.time_float
self.record.append(
(last_line[0], current_time, round(current_time - last_line[0], self.decimal)))
self.status = "paused"
定時任務(wù)的類
class Schedule(threading.Thread):
"""
自定義的定時任務(wù)表,添加第一個定時任務(wù)后就創(chuàng)建一個線程,開始循環(huán)檢查
是否執(zhí)行任務(wù)表中的任務(wù)。
· 調(diào)用stop()來終止該線程。
"""
def __init__(self, *args, **kwargs):
threading.Thread.__init__(self, *args, **kwargs)
self._askToStop = False
self._schedule = [] # 保存定時任務(wù)表
self.status = "initial"
def _get_time(self):
""" 獲取當(dāng)前時間 """
return time.time()
def addTask(self, countDown, func, *args, **kwargs):
"""
在任務(wù)表中增加一項(xiàng)定時任務(wù):在倒計時countDown結(jié)束之后調(diào)用
函數(shù)func,并傳入?yún)?shù)*args和**kwargs。
· 定時任務(wù)只會被執(zhí)行一次,執(zhí)行后就會被從任務(wù)表中刪除。
· 定時任務(wù)只會在倒計時結(jié)束之后被執(zhí)行,但無法保證無延遲。
"""
if self.status == "initial": # 第一次添加定時任務(wù)時創(chuàng)建一個新線程
self.status = "running"
self.start()
task = []
if isinstance(countDown, (int, float)) and countDown > 0:
task.append(self._get_time()+countDown) # 準(zhǔn)備在指定時刻執(zhí)行該任務(wù)
else:
raise ValueError("'countDown' must be a positive int or float.")
if callable(func):
task.append(func)
else:
raise ValueError("'func' must be callable.")
task.append(args) # 保存元組參數(shù)
task.append(kwargs) # 保存字典參數(shù)
self._schedule.append(task)
self._schedule.sort(key=lambda task: task[0]) # 將任務(wù)表按時間戳的大小排序
def _doTask(self):
""" 檢查任務(wù)表中各項(xiàng)任務(wù)的時間,判斷是否要執(zhí)行它。 """
current_time = self._get_time()
i = 0
while i < len(self._schedule): # 遍歷任務(wù)表
task = self._schedule[i]
if task[0] <= current_time:
# 如果該任務(wù)的時間不晚于當(dāng)前時間,就創(chuàng)建一個線程去執(zhí)行該任務(wù),避免阻塞定時器線程
t1 = CreatThread(task[1], *task[2], **task[3])
t1.start()
i += 1
else:
break # 如果該任務(wù)的時間戳大于當(dāng)前時間,就提前結(jié)束遍歷
del self._schedule[:i] # 刪除過時的任務(wù)
def run(self):
""" 線程循環(huán)運(yùn)行的內(nèi)容 """
while not self._askToStop:
self._doTask()
# 結(jié)束時進(jìn)行清理
self.status == "stopped"
return 0
def stop(self):
self._askToStop = True
class CreatThread(threading.Thread):
""" 一個簡單的創(chuàng)建線程的類 """
def __init__(self, func, *args, **kwargs):
threading.Thread.__init__(self)
self.func = func
self.args = args
self.kwargs = kwargs
def run(self):
self.func(*self.args, **self.kwargs)
源代碼:use_time.py
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python3加密解密庫Crypto的RSA加解密和簽名/驗(yàn)簽實(shí)現(xiàn)方法實(shí)例
這篇文章主要介紹了Python3加密解密庫Crypto的RSA加解密和簽名/驗(yàn)簽實(shí)現(xiàn)方法實(shí)例,需要的朋友可以參考下2020-02-02
Python使用asyncio異步時的常見問題總結(jié)
這篇文章主要為大家整理了開發(fā)人員在?Python?中使用?asyncio?時提出的常見問題以及解決方法,文中的示例代碼講解詳細(xì),感興趣的可以學(xué)習(xí)一下2023-04-04
python使用arp欺騙偽造網(wǎng)關(guān)的方法
這篇文章主要介紹了python使用arp欺騙偽造網(wǎng)關(guān)的方法,涉及Python偽造網(wǎng)關(guān)的相關(guān)技巧,需要的朋友可以參考下2015-04-04
Python分析微信好友性別比例和省份城市分布比例的方法示例【基于itchat模塊】
這篇文章主要介紹了Python分析微信好友性別比例和省份城市分布比例的方法,結(jié)合實(shí)例形式分析了Python基于itchat模塊獲取及計算微信好友相關(guān)信息操作技巧,需要的朋友可以參考下2020-05-05

