您好,登錄后才能下訂單哦!
這是我使用python寫的第一個(gè)類(也算是學(xué)習(xí)面向?qū)ο笳Z言以來正式寫的第一個(gè)解耦的類),記錄下改進(jìn)的過程。
分析需求
最初,因?yàn)槭褂胻ime模塊顯示日期時(shí),每次都要設(shè)置時(shí)間字符串的格式,挺麻煩,但還是忍了。
后來,在處理多線程任務(wù)時(shí)需要實(shí)現(xiàn)定時(shí)控制的功能,更麻煩,終于決定自己做一個(gè)解決這些問題的通用代碼(雖然網(wǎng)上有現(xiàn)成的模塊,但親手編寫這部分代碼正好能鍛煉一下我的面向?qū)ο缶幊蹋?/p>
分析框架
剛開始,我計(jì)劃做一個(gè)模仿時(shí)鐘的抽象類,讓它獨(dú)立運(yùn)行在一個(gè)線程中,讓它提供顯示日期、計(jì)時(shí)、設(shè)置定時(shí)任務(wù)的方法……然而由于缺乏規(guī)劃,編程亂糟糟的,這些方法的代碼和變量交雜在一起,難以入目,更難以擴(kuò)展……氣得重構(gòu)代碼,這次把顯示日期、計(jì)時(shí)、設(shè)置定時(shí)任務(wù)三大功能分別抽象成三個(gè)類,相互解耦,各自獨(dú)立運(yùn)行,代碼變得簡(jiǎn)潔多了。
ok,舊代碼就藏在git的歷史記錄里吧,這里貼出重構(gòu)后的代碼。
顯示時(shí)間的類
import time import threading class _Clock: """ 自定義的時(shí)鐘類,用于獲取幾種不同格式的當(dāng)前時(shí)間。 decimal : 設(shè)置time_float的精度,控制其保留幾位小數(shù)。 time_diff : 設(shè)置該時(shí)鐘與UTC+0時(shí)區(qū)的時(shí)差。如果不設(shè)置,會(huì)自動(dòng)采用 本地時(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" # 時(shí)間字符串的格式 @property def time_float(self): """ UTC+0時(shí)區(qū)的時(shí)間戳,精度由self.decimal決定 """ return round(time.time(), self.decimal) @property def time_int(self): """ UTC+0時(shí)區(qū)的時(shí)間戳,精度為秒 """ return int(time.time()) @property def time_tuple(self): """ 本地時(shí)區(qū)的時(shí)間元組 """ 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): """ 本地時(shí)間的格式化字符串 """ return time.strftime(self.time_format, self.time_tuple)
秒表計(jì)時(shí)的類
class Timer(_Clock): """ 自定義的計(jì)時(shí)器,像秒表一樣,可以隨時(shí)查看當(dāng)前計(jì)時(shí)、暫停計(jì)時(shí)、繼續(xù)計(jì)時(shí)。 · 創(chuàng)建一個(gè)計(jì)時(shí)器之后,它就會(huì)開始計(jì)時(shí)。 · 默認(rèn)使用time.time()獲取時(shí)間,精度為毫秒。 · 可以直接調(diào)用_Clock類的方法來獲取當(dāng)前時(shí)間。 """ def __init__(self, *args, **kwargs): _Clock.__init__(self, *args, **kwargs) self.record = [] # 記錄每次使用的 (開始時(shí)刻,暫停時(shí)刻,計(jì)時(shí)時(shí)長(zhǎng)) self.status = "initial" self.go() @property def count(self): """ 當(dāng)前計(jì)時(shí)值 """ 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): """ 開始計(jì)時(shí) """ if self.status != "timing": self.record.append((self.time_float, None, None)) self.status = "timing" def pause(self): """ 暫停計(jì)時(shí) """ # 如果該計(jì)時(shí)器在計(jì)時(shí)中,就暫停它,并計(jì)算這一段的計(jì)時(shí)時(shí)長(zhǎng) 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"
定時(shí)任務(wù)的類
class Schedule(threading.Thread): """ 自定義的定時(shí)任務(wù)表,添加第一個(gè)定時(shí)任務(wù)后就創(chuàng)建一個(gè)線程,開始循環(huán)檢查 是否執(zhí)行任務(wù)表中的任務(wù)。 · 調(diào)用stop()來終止該線程。 """ def __init__(self, *args, **kwargs): threading.Thread.__init__(self, *args, **kwargs) self._askToStop = False self._schedule = [] # 保存定時(shí)任務(wù)表 self.status = "initial" def _get_time(self): """ 獲取當(dāng)前時(shí)間 """ return time.time() def addTask(self, countDown, func, *args, **kwargs): """ 在任務(wù)表中增加一項(xiàng)定時(shí)任務(wù):在倒計(jì)時(shí)countDown結(jié)束之后調(diào)用 函數(shù)func,并傳入?yún)?shù)*args和**kwargs。 · 定時(shí)任務(wù)只會(huì)被執(zhí)行一次,執(zhí)行后就會(huì)被從任務(wù)表中刪除。 · 定時(shí)任務(wù)只會(huì)在倒計(jì)時(shí)結(jié)束之后被執(zhí)行,但無法保證無延遲。 """ if self.status == "initial": # 第一次添加定時(shí)任務(wù)時(shí)創(chuàng)建一個(gè)新線程 self.status = "running" self.start() task = [] if isinstance(countDown, (int, float)) and countDown > 0: task.append(self._get_time()+countDown) # 準(zhǔn)備在指定時(shí)刻執(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ù)表按時(shí)間戳的大小排序 def _doTask(self): """ 檢查任務(wù)表中各項(xiàng)任務(wù)的時(shí)間,判斷是否要執(zhí)行它。 """ current_time = self._get_time() i = 0 while i < len(self._schedule): # 遍歷任務(wù)表 task = self._schedule[i] if task[0] <= current_time: # 如果該任務(wù)的時(shí)間不晚于當(dāng)前時(shí)間,就創(chuàng)建一個(gè)線程去執(zhí)行該任務(wù),避免阻塞定時(shí)器線程 t1 = CreatThread(task[1], *task[2], **task[3]) t1.start() i += 1 else: break # 如果該任務(wù)的時(shí)間戳大于當(dāng)前時(shí)間,就提前結(jié)束遍歷 del self._schedule[:i] # 刪除過時(shí)的任務(wù) def run(self): """ 線程循環(huán)運(yùn)行的內(nèi)容 """ while not self._askToStop: self._doTask() # 結(jié)束時(shí)進(jìn)行清理 self.status == "stopped" return 0 def stop(self): self._askToStop = True class CreatThread(threading.Thread): """ 一個(gè)簡(jiǎn)單的創(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
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。