作爲@offbyone註釋,可以將冗餘處理程序添加到記錄器的同一個實例。 的python docs for logging聲言如下─
「多給getLogger()具有相同的名稱調用將返回一個 參照相同記錄的對象。」
所以我們不需要擔心使實現成爲單例,因爲它已經是了。
不幸的是,同樣是不是真的爲與相同的記錄器實例相關的處理程序。有可以是附加的重複處理程序。
例 -
複製該代碼,並將其保存在main.py
import logging
print 'inside main.py',
print '-'*50
def logger():
print 'initializing logger....'
logPath = '.'
fileName = 'temp'
# configure log formatter
logFormatter = logging.Formatter("%(asctime)s [%(filename)s] [%(funcName)s] [%(levelname)s] [%(lineno)d] %(message)s")
# configure file handler
fileHandler = logging.FileHandler("{0}/{1}.log".format(logPath, fileName))
fileHandler.setFormatter(logFormatter)
# configure stream handler
consoleHandler = logging.StreamHandler()
consoleHandler.setFormatter(logFormatter)
# get the logger instance
logger = logging.getLogger(__name__)
# set the logging level
logger.setLevel(logging.DEBUG)
print 'adding handlers- '
#if not len(logger.handlers):
logger.addHandler(fileHandler)
logger.addHandler(consoleHandler)
print 'logger initialized....\n'
print 'associated handlers - ', len(logger.handlers)
for handler in logger.handlers:
print handler
print
return logger
main_logger = logger()
main_logger.info('utilizing main.py logger.')
print 'exiting main.py',
print '-'*50
和下面的代碼sub.py
print 'inside sub.py',
print '-'*50
print 'importing main.py'
import main
print 'imported main.py'
import logging
print 'getting logger instance in sub'
sub_logger = main.logger()
print 'got logger instance in sub'
sub_logger.info("utilizing sub_logger")
print 'exiting sub.py',
print '-'*50
跑分。該方法的py
[email protected]:~/code/so$ python sub.py
inside sub.py --------------------------------------------------
importing main.py
inside main.py --------------------------------------------------
initializing logger....
adding handlers-
logger initialized....
associated handlers - 2
<logging.FileHandler object at 0x7f7158740c90>
<logging.StreamHandler object at 0x7f7158710b10>
2015-08-04 07:41:01,824 [main.py] [<module>] [INFO] [41] utilizing main.py logger.
exiting main.py --------------------------------------------------
imported main.py
getting logger instance in sub
initializing logger....
adding handlers-
logger initialized....
associated handlers - 4 # <===== 4 handlers (duplicates added)
<logging.FileHandler object at 0x7f7158740c90>
<logging.StreamHandler object at 0x7f7158710b10>
<logging.FileHandler object at 0x7f7158710bd0>
<logging.StreamHandler object at 0x7f7158710c10>
got logger instance in sub
2015-08-04 07:41:01,824 [sub.py] [<module>] [INFO] [10] utilizing sub_logger
2015-08-04 07:41:01,824 [sub.py] [<module>] [INFO] [10] utilizing sub_logger
exiting sub.py --------------------------------------------------
因此多個呼叫返回相同的記錄器添加重複處理程序。現在
,爲您question-
有什麼辦法,我可以檢查,看看是否處理程序已經存在
是的,有是 -
logger.handlers
返回列表所有與給定logger
相關聯的處理程序。
之前將處理程序記錄器的實例,確保不添加重複處理 在main.py,只是取消註釋,上面寫着if not len(logger.handlers):
行和縮進下面兩行properly-
if not len(logger.handlers):
logger.addHandler(fileHandler)
logger.addHandler(consoleHandler)
現在再次運行sub.py
[email protected]:~/code/so$ python sub.py
inside sub.py --------------------------------------------------
importing main.py
inside main.py --------------------------------------------------
initializing logger....
adding handlers-
logger initialized....
associated handlers - 2
<logging.FileHandler object at 0x7fd67a891c90>
<logging.StreamHandler object at 0x7fd67a862b10>
2015-08-04 08:14:45,620 [main.py] [<module>] [INFO] [41] utilizing main.py logger.
exiting main.py --------------------------------------------------
imported main.py
getting logger instance in sub
initializing logger....
adding handlers-
logger initialized....
associated handlers - 2 # <===== Still 2 handlers (no duplicates)
<logging.FileHandler object at 0x7fd67a891c90>
<logging.StreamHandler object at 0x7fd67a862b10>
got logger instance in sub
2015-08-04 08:14:45,620 [sub.py] [<module>] [INFO] [10] utilizing sub_logger
exiting sub.py --------------------------------------------------
此外,如果你想限制要添加到記錄器實例處理程序的類型,你可以不喜歡這個 -
print 'adding handlers- '
# allows to add only one instance of file handler and stream handler
if len(logger.handlers) > 0:
print 'making sure we do not add duplicate handlers'
for handler in logger.handlers:
# add the handlers to the logger
# makes sure no duplicate handlers are added
if not isinstance(handler, logging.FileHandler) and not isinstance(handler, logging.StreamHandler):
logger.addHandler(fileHandler)
print 'added file handler'
logger.addHandler(consoleHandler)
print 'added stream handler'
else:
logger.addHandler(fileHandler)
logger.addHandler(consoleHandler)
print 'added handlers for the first time'
希望這有助於!
編輯:
不幸的是,同樣是不是真的與記錄器的相同實例相關聯 的處理程序。有可以重複 處理程序附加。
事實證明,上述說法並不完全正確。
假設我們已經在主模塊中創建並配置了一個名爲'main_logger'的記錄器(它簡單地配置記錄器,不會返回任何東西)。
# get the logger instance
logger = logging.getLogger("main_logger")
# configuration follows
...
現在一個子模塊,如果我們創建下列命名層次「main_logger.sub_module_logger」孩子記錄儀,我們不需要配置它的子模塊中。只需在命名層次結構中創建記錄器就足夠了。
# get the logger instance
logger = logging.getLogger("main_logger.sub_module_logger")
# no configuration needed
# it inherits the configuration from the parent logger
...
而且它也不會添加重複處理程序。
參考 - Using logging in multiple modules
我想你應該重新評估的答案,因爲mouad的回答(在寫作時)忽略了一個事實的方法,多次調用返回相同的記錄器對象* CAN *添加重複處理。納拉揚做了很好的解釋。 – 2016-02-14 08:11:48