what's log4j actually doing when we turn on or off some log places?

后端 未结 3 1796
礼貌的吻别
礼貌的吻别 2021-01-04 18:02

We know we can config log4j to turn off log on specific places (class or package in Java) via its properties/configuration file. My questions are followed:

  1. wha
3条回答
  •  悲哀的现实
    2021-01-04 18:40

    Yes, the log statements will still be executed. This is why it's a good pattern to check the log level first: something like

    if (log.isInfoEnabled()) {
        log.info("My big long info string: " + someMessage);
    }
    

    This is to keep from reallocating space for the info String when the log level does not support INFO statements.

    It's not anything like #ifdef - #ifdef is a compiler directive, whereas Log4J configurations are processed at runtime.

    Edit: I hate getting downmodded due to ignorance, so here is one article backing up my answer.

    From http://surguy.net/articles/removing-log-messages.xml:

    In Log4J, if you log a message at DEBUG level, and the current Appender is set to only log messages of INFO level and above, then the message will not be displayed. The performance penalty for calling the log method itself is minimal - a few nanoseconds. However, it may take longer to evaluate the arguments to the log method. For example:

    logger.debug("The large object is "+largeObject.toString());

    Evaluating largeObject.toString() may be slow, and it is evaluated before the call to the logger, so the logger cannot prevent it being evaluated, even though it will not be used.

    Edit 2: from the log4j manual itself (http://logging.apache.org/log4j/1.2/manual.html):

    The user should be aware of the following performance issues.

    1. Logging performance when logging is turned off. When logging is turned off entirely or just for a set of levels, the cost of a log request consists of a method invocation plus an integer comparison. On a 233 MHz Pentium II machine this cost is typically in the 5 to 50 nanosecond range.

      However, The method invocation involves the "hidden" cost of parameter construction.

      For example, for some logger cat, writing,

       logger.debug("Entry number: " + i + " is " + String.valueOf(entry[i]));
      

      incurs the cost of constructing the message parameter, i.e. converting both integer i and entry[i] to a String, and concatenating intermediate strings, regardless of whether the message will be logged or not. This cost of parameter construction can be quite high and it depends on the size of the parameters involved.

      To avoid the parameter construction cost write:

      if(logger.isDebugEnabled() {
        logger.debug("Entry number: " + i + " is " + String.valueOf(entry[i]));
      }
      

      This will not incur the cost of parameter construction if debugging is disabled. On the other hand, if the logger is debug-enabled, it will incur twice the cost of evaluating whether the logger is enabled or not: once in debugEnabled and once in debug. This is an insignificant overhead because evaluating a logger takes about 1% of the time it takes to actually log.

提交回复
热议问题