溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

python自定義時(shí)鐘類、定時(shí)任務(wù)類

發(fā)布時(shí)間:2020-08-25 04:13:17 來源:腳本之家 閱讀:171 作者:威爾阿威 欄目:開發(fā)技術(shù)

這是我使用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í)有所幫助,也希望大家多多支持億速云。

向AI問一下細(xì)節(jié)

免責(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)容。

AI