python logging模塊的作用及應用場景_Python常用模塊功能簡介(三)logging

logging基本介紹

先介紹一下我們為什么要使用日志,平常我們編寫程序為了驗證程序運行與debug,通常會使用print函數來對一些中間結果進行輸出驗證,在驗證成功后再將print語句注釋或刪除掉。這樣做在小型程序中還比較靈活,但是對于大型項目來說,就十分繁瑣了----->所以使用日志log就很自然了,日志可以調整日志級別,來決定我們是否輸出對應級別的日志,同時還可以將日志導入文件記錄下來。

再介紹一下logging中的log級別:

Level

Numeric Value

logging.CRITICAL

50

logging.ERROR

40

logging.WARNING

30

logging.INFO

20

logging.DEBUG

10

實際上這些level都是整數值,可由如type(logging.info)驗證為int類型。

模塊級的使用方法

實際上logging是一個package而不是module,以下對于源碼的討論都是在logging的__init__.py文件中的。

logging模塊的模塊級使用方法就是使用一些模塊級接口函數。而且還有一個比較重要的是對日志的輸出形式和輸出目的地等進行設置。

接口函數也是對應日志級別而輸出信息的:

logging.debug(msg)

logging.info(msg)

logging.warning(msg)

logging.error(msg)

logging.critical(msg)

這幾個函數除了日志級別上的區別,其實都是使用默認的root logger來對信息進行log的,它是處于日志器層級關系最頂層的日志器,且該實例是以單例模式存在的。見源碼:

def info(msg, *args, **kwargs):

if len(root.handlers) == 0:

basicConfig()

root.info(msg, *args, **kwargs)

這里可以見到在logging模塊的info函數中:(1)首先進行了一個對于root logger的handlers屬性的長度判斷是否調用basicConfig函數。(2)之后是調用root logger的info函數來實現功能的。

這里對于第(2)點我們進一下探尋:

root = RootLogger(WARNING)

在logging的源碼中可以看到如上語句,即我們將logging模塊import后,其實已經默認的創建了一個root logger對象,并且logging自己維護有一個Logger實例的hierarchy(通過Manager實例對象,它和root logger一樣也是單例的使用模式),我們自己創建的logger實例對象,都是是root logger的孩子。

日志的設置

對于日志的設置,我們是使用logging.basicConfig(**kwargs)函數,它是對root logger進行設置的,一般使用較多的關鍵字參數如下:

level 決定root logger的日志級別。

format 它是日志格式字符串,指定日志輸出的字段信息,如日期,日志級別,文件名,當然還有我們的msg信息等等。

datefmt 決定日期字段的輸出格式。

filename 日志信息的輸出目的地文件,不指定時默認輸出到控制臺。

filemode 目的地文件的打開模式,不指定默認為"a"。

對于logging.basicConfig函數有一點需要注意:我們不能在該函數前使用任何模塊級日志輸出函數如logging.info、logging.error,因為它們會調用一個不帶參的basicConfig函數,使得logging.basicConfig函數失效。見源碼(由于代碼過多,建議參考注釋進行閱讀):

def basicConfig(**kwargs):

_acquireLock()

try:

#這里由于不帶參調用basicConifg,

#而root.handlers默認為空列表

#在Logger定義中可見self.handlers被設為[],

#而默認的root實例在創建時只指定了log級別

#所以if條件必然通過

if len(root.handlers) == 0:

#由于不帶參,所以handlers必為None

handlers = kwargs.pop("handlers", None)

if handlers is None:

#這里由于不帶參,所以即是handlers為None

#通過上面的if判斷,但kwargs同樣為None,

#所以該if不通過

if "stream" in kwargs and "filename" in kwargs:

raise ValueError("'stream' and 'filename' should not be "

"specified together")

else:

if "stream" in kwargs or "filename" in kwargs:

raise ValueError("'stream' or 'filename' should not be "

"specified together with 'handlers'")

#這里由于handlers為None通過if判斷繼續執行

if handlers is None:

filename = kwargs.pop("filename", None)

mode = kwargs.pop("filemode", 'a')

if filename:

h = FileHandler(filename, mode)

#不帶參,kwargs為None,所以filename

#在上面的執行語句的返回值為None,所以

#執行這個else分支

else:

stream = kwargs.pop("stream", None)

h = StreamHandler(stream)

#注意這里,十分重要,可見handlers終于不為None

#被賦予了一個列表,該列表有一個元素h

handlers = [h]

dfs = kwargs.pop("datefmt", None)

style = kwargs.pop("style", '%')

if style not in _STYLES:

raise ValueError('Style must be one of: %s' % ','.join(

_STYLES.keys()))

fs = kwargs.pop("format", _STYLES[style][1])

fmt = Formatter(fs, dfs, style)

#再看這里,十分重要

for h in handlers:

#這個無所謂,就是對format進行默認設置

if h.formatter is None:

h.setFormatter(fmt)

#這里最為關鍵,可見root.addHandler(h)函數

#會把h添加進root.handlers列表中,那么很顯然

#root.handlers不再是一個空列表

root.addHandler(h)

level = kwargs.pop("level", None)

if level is not None:

root.setLevel(level)

if kwargs:

keys = ', '.join(kwargs.keys())

raise ValueError('Unrecognised argument(s): %s' % keys)

finally:

_releaseLock()

所以即是不帶參調用basicConfig(),但是經過其函數體執行,root.handlers的列表長度會不為0,所以之后再調用logging.basicConifg函數時,對root.handlers判斷,就會因此而直接略過函數體中try部分(主要部分),直接執行finally,沒有進行任何設置。(并且很關鍵的basicConfig函數就是對root logger的handlers進行設置)

對象級使用

在logging模塊中logger對象從來都不是直接實例化,而是通過一個模塊級借口完成:logging.getLogger(name=None),注意我們創建的logger都是root logger的子類。而通過我們自己創建的logger對象,使用日志記錄也是和模塊級接口一樣的:

logger.debug(msg)

logger.info(msg)

logger.warning(msg)

logger.error(msg)

logger.critical(msg)

同樣對于日志格式的設置也是通過logging.basicConfig函數完成的,雖然該函數是對root logger實例對象的日志格式設置,但由于Logger類方法的特殊實現,其實是能夠沿用該設置的。

并且對于對象級接口,如logger.info函數:

def info(self, msg, *args, **kwargs):

if self.isEnabledFor(INFO):

self._log(INFO, msg, args, **kwargs)

可見其中沒有對basicConfig函數的調用,所以也就沒有修改root.handlers列表,即不會發生上文的logging.basciConfig函數失效的問題。

Logger類及basicConfig沿用問題(無興趣可略)

上文提到了由于Logger類中方法的特殊實現,使得之后實例化的logger對象在進行日志輸出記錄時也能沿用root logger的basicConfig設定。而這背后的機制到底是怎樣的呢?先看:

class RootLogger(Logger):

def __init__(self, level):

"""

Initialize the logger with the name "root".

"""

Logger.__init__(self, "root", level)

_loggerClass = Logger

.....

#由上面代碼可見RootLogger其實就是調用了Logger類

root = RootLogger(WARNING)

#這里是對Logger類綁定兩個類級屬性

#把root實例作為Logger類的一個root屬性

Logger.root = root

#把root實例作為參數傳入Manager類,

#并用返回的實例定義Logger類的manager屬性

#并且Manager類只會被實例化這一次

Logger.manager = Manager(Logger.root)

以上源碼中出現的Manager類很重要,在我們的logging.getLogger函數中:

def getLogger(name=None):

"""

Return a logger with the specified name,

creating it if necessary.

If no name is specified, return the root logger.

"""

if name:

#結合上文的代碼,我們使用root實例作為參數

#傳入Manager類定義了Logger.manager屬性,

#此處使用manager實例所有的getLogger方法

#來具體實現模塊級的getLogger方法

return Logger.manager.getLogger(name)

else:

return root

小結:

Logger.manager屬性是Manager(root)的實例,而logging.getLogger其實就是由Logger.manager對象調用它的getLogger方法。

再轉到Manager類的定義中,來分析Manager(Logger.root)創建manager實例對象和用logger.getLogger時具體發生了什么:

class Manager(object):

"""

holds the hierarchy of loggers.

"""

#Logger.manager = Manager(Logger.root)發生的事情

def __init__(self, rootnode):

"""

Initialize the manager with the root node of the logger hierarchy.

"""

#結合上文Logger.manager = Manager(Logger.root)

#可見我們使用root logger作為rootnode,并把它賦予

#Manager類的self.root屬性

self.root = rootnode

self.disable = 0

self.emittedNoHandlerWarning = False

#self.loggerDict維護logger hierarchy中的loggers

self.loggerDict = {}

self.loggerClass = None

self.logRecordFactory = None

#使用logging.getLogger時發生的事情,Logger.manager屬性

#也就是使用root logger作為參數的Manager實例調用該實例

#所屬Manager類的getLogger方法

def getLogger(self, name):

#這里創建一個值為None的rv

rv = None

#不用理會,只是判斷name值是否為str

if not isinstance(name, str):

raise TypeError('A logger name must be a string')

_acquireLock()

try:

#這里由于是使用Logger.manager實例來調用

#getLogger方法,而Logger.manager實例的

#self.loggerDict初始值為空,所以在第一

#次使用getLogger方法時,這個判斷不通過

#就算之后再次調用logging.getLogger(name)

#也要看name是否在loggerDict中

if name in self.loggerDict:

rv = self.loggerDict[name]

if isinstance(rv, PlaceHolder):

ph = rv

rv = (self.loggerClass or _loggerClass)(name)

rv.manager = self

self.loggerDict[name] = rv

self._fixupChildren(ph, rv)

self._fixupParents(rv)

#所以直接轉到此處

else:

#這里self.loggerClass初始為空,使用

#_loggerClass其實就是Logger類實例化

#一個logger對象附于rv

rv = (self.loggerClass or _loggerClass)(name)

#這里再把self也就是Logger.manager賦予

#rv.manager,并未創建新的Manager實例

rv.manager = self

#把rv加入loggerDict中,name:rv鍵值對

self.loggerDict[name] = rv

#設定當前logger實例的parent

self._fixupParents(rv)

finally:

_releaseLock()

#返回rv,也就是創建的logger實例對象

return rv

大概介紹了Manager的部分源碼,我們回到最初的問題,為何我們自己創建的logger實例能沿用root logger的basicConfig設定,再看logger.info方法的源碼:

#logger實例的info方法

def info(self, msg, *args, **kwargs):

#對比INFO日志級別和當前logger實例的日志級別來決定是否進行log

if self.isEnabledFor(INFO):

#所以這里其實是調用了logger實例的底層_log方法

self._log(INFO, msg, args, **kwargs)

def _log(self, level, msg, args, exc_info=None, extra=None, stack_info=False):

"""

Low-level logging routine which

creates a LogRecord and then calls

all the handlers of this logger to

handle the record.

"""

#為避免繁瑣,把一些簡單使用時不會碰到的代碼略過

...

#生成log record

record = self.makeRecord(self.name, level, fn, lno, msg, args,

exc_info, func, extra, sinfo)

#調用當前logger實例的handle方法來處理log record

self.handle(record)

def handle(self, record):

if (not self.disabled) and self.filter(record):

#可見logger類的handle方法又調用了該類的

#callHandlers方法來處理record

self.callHandlers(record)

#注意看源碼中自帶注釋對于callHandlers方法解釋,其中

#很重要的一點就是,該函數會loop through該logger乃至

#該logger的parent,parent的parent一直到root logger

#中的handlers,注意我們在前文中說了,basicConfig其實

#就是對root logger的handlers進行設置

def callHandlers(self, record):

"""

Pass a record to all relevant handlers.

Loop through all handlers for this logger

and its parents in the logger hierarchy.

If no handler was found, output a one-off error

message to sys.stderr. Stop searching up

the hierarchy whenever a logger with the

"propagate" attribute set to zero is found - that

will be the last logger whose handlers are called.

"""

#把當前logger實例賦予c

c = self

found = 0

#這個while一定可以進去,因為c是一個logger對象,不為None

while c:

#對于我們創建的logger,它的handlers初始值是空列表

#所以這個for一開始進不去,對于我們的簡單使用場景

#使用了logging.basicConfig(...),然后創建自己的

#logger=logging.getLogger("mylogger"),當前

#logger的parent肯定是root logger,而前面的

#basicConfig對root logger的handlers設置了,

#所以root logger的handlers不為空,可以進入

for hdlr in c.handlers:

found = found + 1

if record.levelno >= hdlr.level:

#使用handlers中的handler對

#log record進行handle處理

hdlr.handle(record)

#這個也進不去,logger實例的propagate都是默認True

if not c.propagate:

c = None #break out

#只能進這里了

else:

#把當前logger實例的parent賦予c

#然后繼續循環,如果handlers仍然為空

#則繼續把parent賦予c直到root logger

c = c.parent

...

簡而言之,我們自己創建的logger能使用parent(Manager實例所維持的一個hierarchy)的handlers對log信息進行handle,而向上追溯的祖先就是root logger,我們已經通過basicConfig的話,實質上就是對它的handlers進行了設置,所以在這個hierarchy中的后代便能享用。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/534686.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/534686.shtml
英文地址,請注明出處:http://en.pswp.cn/news/534686.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

計算機科技文獻中 CAM,計算機輔助設計、制造(CAD、CAM)和《機械制圖》 課程的結合、探索與實踐研究...

王建臣摘要:隨著我國科學技術的迅猛發展,計算機技術也隨之得到大范圍推廣,并迎來跨越式的發展,在機械行業中,管理、設計、制造都已經逐漸換裝計算機來進行。計算機機械制圖作為機械專業的基礎課程,肩負著培…

docker linux 快速開窗口_Linux搭建docker環境并簡單實用

記錄生活:配置阿里云鏡像源docker YUM[rootcentos-linux ~]# sudo yum install -y yum-utils device-mapper-persistent-data lvm2[rootcentos-linux ~]# sudo yum-config-manager --add-repo https://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce.repo安裝…

清華計算機接口原理,微機原理與接口技術課后習題答案清華大學

微機原理與接口技術課后習題答案清華大學 微機原理與接口技術課后部分習題參考答案 第一章 2. 第 3項任務,根據狀態標志位的狀態決定轉移方向。 3. 程序存儲是將要執行的程序的全部指令存儲到存儲器中,程序控制指程序開始執行后,通過指令流控…

vf求計算機系統當前日期的年份數,計算機二級VF常用函數列表

數值函數:1.絕對值和符號函數格式:ABS()SIGN()例如:ABS(-5)5,ABS(4)4,SIGN(8)1,SIGN(-3)-1,SIGN(0)02.求平方根表達式格式:SQRT()例如:SQRT(16)4,它與開二分之一次方等同。3.圓周率函數格式:PI()4.求整數函…

element走馬燈自動_詳細element-ui的走馬燈carousel輕松實現自適應全屏banner詳細過程...

div部分CSS部分.bannerImg{width: 100%;height: inherit;min-height: 600px;min-width: 1400px;}vue.JS部分var vm new Vue({el : #apptwo,data : {bannerHeight:"",BannerImg:["${request.contextPath}/statics/BCHimg/b6.png","${request.contextP…

4n35光耦引腳圖_光耦繼電器的主要特點以及輸入特性!先進光半導體

光耦合器是一種電光電轉換設備,可將電信號作為介質傳輸。它由兩部分組成:光源和光接收器。發光源和光接收器被組裝在同一氣密殼體中,并通過透明絕緣體彼此隔開。發光源的引腳為輸入端子,光接收器的引腳為輸出端子,公共…

系統流暢度測試軟件,如何通過FPS顯示快速測試自己手機流暢度

對于手機流暢度測試除了系統自帶的功能能測試手機流暢度之外,那么就沒有其它方法能測試手機流暢度了么?那么接下來由小編為大家推薦一款小工具FPS顯示(FPS Meter),你可以通過它快速的測試自己手機流暢度,接下來一起看看吧&#xf…

中軟國際軟件測試培訓中心,中軟國際準員工培養計劃C++開發/軟件測試方向開班典禮...

2010年8月31日上午9點半,中軟國際準員工培養計劃—C開發/軟件測試方向開班典禮在無錫ETC隆重舉行。開班典禮在熱烈的掌聲中拉開了序幕。典禮由教務部經理陳晨老師主持,首先她對近20名學員的到來表示歡迎,并對本次學習班順利開班表示祝賀。中軟…

django filter查詢多選_動態filter查詢數據Django實現方法

這是我在學習Django時看到的關于動態filter查詢數據Django實現方法。當時很受用,解決了我在工作中遇到的一個問題。可能有朋友會需要,轉來玩蛇網python學習平臺和大家一同分享下。在平時用Djangoa工作時,需要處理像是對多個字段進行查詢找結果…

二本考北航計算機經歷,我(來自二本學校)考上北航的一些經歷

大家很想知道二本考上是怎麼考上北航的吧!我就把我的一點經歷告訴大家吧。本人由于高考發揮的不好來到了一所很不入流的二本學校。所以我決定利用考研的機會進入理想大學。跟大家一樣我也是從大三下學期開始準備考研的,由于本人英語還可以,所…

pytorch 命令行運行_PyTorch簡介與相關安裝

PyTorch簡介PyTorch的前身是Torch,Torch是一個有大量機器學習算法支持的科學計算框架,靈活度很高,是一個與NumPy類似的張量(Tensor)操作庫。但是Torch采用的是小眾的編程語言Lua,因此流行度很低,于是就有了PyTorch的出…

軟件測試用例分享ppt,分享:測試用例設計方法.ppt

分享:測試用例設計方法.ppt還剩122頁未讀,繼續閱讀下載文檔到電腦,馬上遠離加班熬夜!親,很抱歉,此頁已超出免費預覽范圍啦!如果喜歡就下載吧,價低環保!內容要點&#xff…

python 命名空間沖突_python-命名空間

通俗的來說,Python中所謂的命名空間可以理解為一個容器。在這個容器中可以裝許多標識符。不同容器中的同名的標識符是不會相互沖突的。理解python的命名空間需要掌握三條規則:第一,賦值(包括顯式賦值和隱式賦值)產生標識符,賦值的…

修改tomcat服務器圖標,修改tomcat小貓圖標,設置項目的favicon圖標

JAVA設計模式之模板模式在閻宏博士的一書中開頭是這樣描述模板方法(Template Method)模式的: 模板方法模式是類的行為模式.準備一個抽象類,將部分邏輯以具體方法以及具體構造函數的形式 ...從scheduler is shutted down看程序員的英文水平我有個windows服務程序,今天重點在測試…

golang 安全的tcp server_Go 語言使用 TCP_NODELAY 控制發包流量

編寫健壯且高性能的網絡服務需要付出大量的努力。提高服務性能的方式有很多種,比如優化應用層的代碼,更進一步,還可以看看垃圾回收器,操作系統,網絡傳輸,以及部署我們服務的硬件是否有優化空間。TCP/IP 協議…

react取消所有請求_react 組件關閉后怎么消除還在進行中的ajax

把你的請求做成可以取消的, 這里的取消不是取消發送。 因為請求已經發送了,沒有辦法終止的。所謂的取消其實就是取消回調函數, react官方給出了一種最佳實踐。const makeCancelable (promise) > { let hasCanceled_ false; const wrap…

css 漣漪,CSS3水波漣漪動畫定位樣式如何制作

CSS3水波漣漪動畫定位樣式如何制作寶劍鋒從磨礪出,梅花香自苦寒來。以下是小編為大家搜索整理的CSS3水波漣漪動畫定位樣式如何制作,希望能給大家帶來幫助!更多精彩內容請及時關注我們應屆畢業生考試網!先上效果圖:教程本動畫需要用到的主要屬性:animatio…

python基礎實訓_python基礎實踐(三)

-*-列表是新手可直接使用的最強大的python功能之一,它融合了眾多重要的編程概念。-*-# -*- coding:utf-8 -*-# Author:sweeping-monkQuestion_1 "什么是列表?"print(Question_1)smg "列表由一系列按特定順序排列的元素組成。你可以創建…

python axis 0 1_python pandas 中axis值0 1怎么分行還是列

axis的重點在于方向,而不是行和列。1表示橫軸,方向從左到右;0表示縱軸,方向從上到下。即axis1為橫向,axis0為縱向,而不是行和列,具體到各種用法而言也是如此。當axis1時,如果是求平均…

ajax php接收不到數據庫,PHP更新MySQL數據庫與AJAX調用沒有做任何事情

我已經測試過,發現正確的數據被發送,但PHP更新數據庫中的字段即處理更新無法正常工作。發生的一切就是我在條件中得到了else響應。我需要根據用戶輸入是什么來更新數據庫。就像我說的,我得到的回應是else回應。$youruname $_POST[youruname]…