Warning: file_get_contents(/data/phpspider/zhask/data//catemap/2/python/311.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181

Warning: file_get_contents(/data/phpspider/zhask/data//catemap/8/python-3.x/17.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Python 重复的日志输出,但没有重复的处理程序_Python_Python 3.x_Logging_Python Logging - Fatal编程技术网

Python 重复的日志输出,但没有重复的处理程序

Python 重复的日志输出,但没有重复的处理程序,python,python-3.x,logging,python-logging,Python,Python 3.x,Logging,Python Logging,我从一个只连接了1个处理程序的记录器获得了重复的日志输出 我如何调试这个 代码(应可复制和粘贴): 默认情况下,记录器将事件记录到其祖先的处理程序中a_logger可能只有一个处理程序,但它的父级root logger也有一个处理程序(实际上是同一个处理程序)a_logger.info('hi')由两个处理程序处理(实际上是同一个处理程序处理两次) 您不需要将相同的处理程序附加到每个记录器。将其附加到根记录器就足够了。为什么要将同一个处理程序添加到四个记录器?我明白了,所以除了检查记录器本身之外

我从一个只连接了1个处理程序的记录器获得了重复的日志输出

我如何调试这个

代码(应可复制和粘贴):

默认情况下,记录器将事件记录到其祖先的处理程序中
a_logger
可能只有一个处理程序,但它的父级root logger也有一个处理程序(实际上是同一个处理程序)
a_logger.info('hi')
由两个处理程序处理(实际上是同一个处理程序处理两次)


您不需要将相同的处理程序附加到每个记录器。将其附加到根记录器就足够了。

为什么要将同一个处理程序添加到四个记录器?我明白了,所以除了检查记录器本身之外,我还需要检查子记录器的父级是否有处理程序?@shadowtalker:您不需要手动将处理程序附加到每个记录器。您已将其连接到根记录器。够了。此代码将用于除根记录器以外的记录器(可能不是任何子记录器的父级)。@shadowtalker:你是说根记录器可能不是任何子记录器的父级吗?根记录器是所有其他记录器的祖先。如果您说根记录器可能不是您添加处理程序的记录器之一,那么您将遇到一些问题,例如在将处理程序添加到记录器的后代之后,将处理程序添加到记录器时会发生什么情况。@shadowtalker:我不知道。无论哪种方式都有问题。(一般来说,我真的不喜欢
日志记录
模块设计。)
import io
import logging
import os
import sys
from typing import Optional


DEFAULT_LOG_LEVEL = logging.INFO


def get_sub_loggers():
    return [
        logging.getLogger('a'),
        logging.getLogger('b'),
        logging.getLogger('c')
    ]


def safe_add_handler(logger: logging.Logger, h: logging.Handler) -> None:
    if h not in logger.handlers:
        logger.addHandler(h)


def safe_add_filter(logger: logging.Logger, f: logging.Filter) -> None:
    if f not in logger.filters:
        logger.addFilter(f)


def configure_logging(main_logger: logging.Logger,
                      log_level = DEFAULT_LOG_LEVEL,
                      log_file = None,
                      fmt: Optional[str] = None,
                      force_add_handler: bool = False) -> None:
    main_logger.setLevel(log_level)
    logging.captureWarnings(True)  # send all warnings to be logged

    if force_add_handler or not main_logger.hasHandlers():
        if log_file is None or log_file == '-':
            log_file = sys.stderr

        if isinstance(log_file, (str, os.PathLike)):
            handler = logging.FileHandler(log_file)
        else:
            handler = logging.StreamHandler(log_file)

        handler.setFormatter(logging.Formatter(fmt if fmt is not None else logging.BASIC_FORMAT))

        main_logger.addHandler(handler)
        log_output_location = handler.stream.name if isinstance(handler, logging.StreamHandler) else handler.baseFilename
        main_logger.info('Writing logs to {}'.format(log_output_location))
    else:
        main_logger.info('Using existing log handlers')

    main_logger.info('Using log level {}'.format(log_level))

    for logger in get_sub_loggers():
        logger.setLevel(log_level)
        for h in main_logger.handlers:
            safe_add_handler(logger, h)
        for f in main_logger.filters:
            safe_add_filter(logger, f)
        main_logger.debug('Configured logger {}'.format(logger.name))

    main_logger.info('Configured submodule loggers.')


configure_logging(logging.getLogger())

a_logger = logging.getLogger('a')
assert len(a_logger.handlers) == 1

# Should only output "hi" once, but it does it twice
a_logger.info('hi')