欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2021年02月22日 14:40:52   作者:威爾阿威  
這篇文章主要為大家詳細(xì)介紹了Python自定義時(shí)鐘類、定時(shí)任務(wù)類,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

這是我使用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)行,代碼變得簡潔多了。
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í)長)
  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í)長
  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è)簡單的創(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)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Python3加密解密庫Crypto的RSA加解密和簽名/驗(yàn)簽實(shí)現(xiàn)方法實(shí)例

    Python3加密解密庫Crypto的RSA加解密和簽名/驗(yàn)簽實(shí)現(xiàn)方法實(shí)例

    這篇文章主要介紹了Python3加密解密庫Crypto的RSA加解密和簽名/驗(yàn)簽實(shí)現(xiàn)方法實(shí)例,需要的朋友可以參考下
    2020-02-02
  • pyside寫ui界面入門示例

    pyside寫ui界面入門示例

    PySide 是一個(gè)python綁定的跨平臺(tái)GUI Qt庫。目前,支持Python的Qt庫有兩個(gè)PyQt和PySide。下面寫一個(gè)簡單的hello程序?qū)W習(xí)他的使用方法
    2014-01-01
  • Python使用asyncio異步時(shí)的常見問題總結(jié)

    Python使用asyncio異步時(shí)的常見問題總結(jié)

    這篇文章主要為大家整理了開發(fā)人員在?Python?中使用?asyncio?時(shí)提出的常見問題以及解決方法,文中的示例代碼講解詳細(xì),感興趣的可以學(xué)習(xí)一下
    2023-04-04
  • python如何調(diào)用百度識(shí)圖api

    python如何調(diào)用百度識(shí)圖api

    這篇文章主要介紹了python如何調(diào)用百度識(shí)圖api,幫助大家更好的理解和使用python,感興趣的朋友可以了解下
    2020-09-09
  • Python中re模塊的元字符使用小結(jié)

    Python中re模塊的元字符使用小結(jié)

    元字符是正則表達(dá)式中具有特殊意義的專用字符,本文主要介紹了Python中re模塊的元字符使用小結(jié),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-04-04
  • python串口讀取數(shù)據(jù)的實(shí)例

    python串口讀取數(shù)據(jù)的實(shí)例

    這篇文章主要介紹了python串口讀取數(shù)據(jù)的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-09-09
  • python使用arp欺騙偽造網(wǎng)關(guān)的方法

    python使用arp欺騙偽造網(wǎng)關(guān)的方法

    這篇文章主要介紹了python使用arp欺騙偽造網(wǎng)關(guān)的方法,涉及Python偽造網(wǎng)關(guān)的相關(guān)技巧,需要的朋友可以參考下
    2015-04-04
  • Python?list列表查找元素詳情

    Python?list列表查找元素詳情

    這篇文章主要介紹了Python?list列表查找元素詳情,Python?列表(list)提供了?index和count方法,它們都可以用來查找元素,文章圍繞主題的相關(guān)資料展開詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)價(jià)值,需要的朋友可以參考一下
    2022-06-06
  • Python分析微信好友性別比例和省份城市分布比例的方法示例【基于itchat模塊】

    Python分析微信好友性別比例和省份城市分布比例的方法示例【基于itchat模塊】

    這篇文章主要介紹了Python分析微信好友性別比例和省份城市分布比例的方法,結(jié)合實(shí)例形式分析了Python基于itchat模塊獲取及計(jì)算微信好友相關(guān)信息操作技巧,需要的朋友可以參考下
    2020-05-05
  • Python jieba結(jié)巴分詞原理及用法解析

    Python jieba結(jié)巴分詞原理及用法解析

    這篇文章主要介紹了Python jieba結(jié)巴分詞原理及用法解析,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-11-11

最新評(píng)論