您好,登錄后才能下訂單哦!
這篇文章給大家介紹python中怎么利用TimedRotatingFileHandler按時間切割日志,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
通過TimedRotatingFileHandler按時間切割日志
線上跑了一個定時腳本,每天生成的日志文件都寫在了一個文件中。但是日志信息不可能輸出到單一的一個文件中。
原因有二:1.日志文件越來越大會影響系統的性能。2.日志文件格式不夠清晰,比如我想看今天的日志,不太方便找到的今天的日志信息(即使對日志輸出做了時間提示)
通過設置 TimedRotatingFileHandler
進行日志按周(W)、天(D)、時(H)、分(M)、秒(S)切割。
先看一個簡單例子:
import time import logging import os from logging import handlers def _logging(**kwargs): level = kwargs.pop('level', None) filename = kwargs.pop('filename', None) datefmt = kwargs.pop('datefmt', None) format = kwargs.pop('format', None) if level is None: level = logging.DEBUG if filename is None: filename = 'default.log' if datefmt is None: datefmt = '%Y-%m-%d %H:%M:%S' if format is None: format = '%(asctime)s [%(module)s] %(levelname)s [%(lineno)d] %(message)s' log = logging.getLogger(filename) format_str = logging.Formatter(format, datefmt) # backupCount 保存日志的數量,過期自動刪除 # when 按什么日期格式切分(這里方便測試使用的秒) th = handlers.TimedRotatingFileHandler(filename=filename, when='S', backupCount=3, encoding='utf-8') th.setFormatter(format_str) th.setLevel(logging.INFO) log.addHandler(th) log.setLevel(level) return log os.makedirs("./logs", exist_ok=True) logger = _logging(filename='./logs/default.log') if __name__ == '__main__': while True: time.sleep(0.1) logger.info('哈哈哈')
結果如下:
上述代碼可以正常運行,而且也可以生成固定的日志個數,但是有一個問題,生成的日志文件格式是你的 文件名+時間 的格式,沒有設置時間的話默認設置到了秒(這里是按秒切割)
修改日志格式后綴名稱:
# 在上述代碼中加入 def namer(filename): return filename.split('default.') th.namer = namer # 設置為S,默認的suffix為 Y-%m-%d_%H-%M-%S th.suffix = "%Y-%m-%d_%H-%M-%S.log" # 為了看的更視覺效果,可以顯示在控制臺答應 cmd = logging.StreamHandler() cmd.setFormatter(format_str) cmd.setLevel(level) log.addHandler(cmd)
運行結果:
名字好像可以了,但是日志好像沒有起到自動刪除的目的啊,而且也沒在之前的log文件夾了。
來看看源碼:
def getFilesToDelete(self): """ Determine the files to delete when rolling over. More specific than the earlier method, which just used glob.glob(). """ dirName, baseName = os.path.split(self.baseFilename) fileNames = os.listdir(dirName) result = [] prefix = baseName + "." plen = len(prefix) for fileName in fileNames: if fileName[:plen] == prefix: suffix = fileName[plen:] if self.extMatch.match(suffix): result.append(os.path.join(dirName, fileName)) if len(result) < self.backupCount: result = [] else: result.sort() result = result[:len(result) - self.backupCount] return result
這是它的刪除邏輯,關鍵是通過 . 前面的字段判斷是否重復,當有特定的重復數后開始刪除。
所以問題來了,要么自己去重寫源碼,要么就只能用 default.日期.log 這種格式了。
附上平時使用的日志代碼
import logging import os from logging import handlers def _logging(**kwargs): level = kwargs.pop('level', None) filename = kwargs.pop('filename', None) datefmt = kwargs.pop('datefmt', None) format = kwargs.pop('format', None) if level is None: level = logging.DEBUG if filename is None: filename = 'default.log' if datefmt is None: datefmt = '%Y-%m-%d %H:%M:%S' if format is None: format = '%(asctime)s [%(module)s] %(levelname)s [%(lineno)d] %(message)s' log = logging.getLogger(filename) format_str = logging.Formatter(format, datefmt) def namer(filename): return filename.split('default.')[1] # cmd = logging.StreamHandler() # cmd.setFormatter(format_str) # cmd.setLevel(level) # log.addHandler(cmd) os.makedirs("./debug/logs", exist_ok=True) th_debug = handlers.TimedRotatingFileHandler(filename="./debug/" + filename, when='D', backupCount=3, encoding='utf-8') # th_debug.namer = namer th_debug.suffix = "%Y-%m-%d.log" th_debug.setFormatter(format_str) th_debug.setLevel(logging.DEBUG) log.addHandler(th_debug) th = handlers.TimedRotatingFileHandler(filename=filename, when='D', backupCount=3, encoding='utf-8') # th.namer = namer th.suffix = "%Y-%m-%d.log" th.setFormatter(format_str) th.setLevel(logging.INFO) log.addHandler(th) log.setLevel(level) return log os.makedirs('./logs', exist_ok=True) logger = _logging(filename='./logs/default')
關于python中怎么利用TimedRotatingFileHandler按時間切割日志就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。