No log4j, a verificação de isDebugEnabled antes do registro melhora o desempenho?


207

Estou usando o Log4J no meu aplicativo para log. Anteriormente, eu estava usando a chamada de depuração como:

Opção 1:

logger.debug("some debug text");

mas alguns links sugerem que é melhor verificar isDebugEnabled()primeiro, como:

Opção 2:

boolean debugEnabled = logger.isDebugEnabled();
if (debugEnabled) {
    logger.debug("some debug text");
}

Portanto, minha pergunta é " A opção 2 melhora o desempenho de alguma maneira? ".

Porque, em qualquer caso, a estrutura Log4J tem a mesma verificação para debugEnabled. Para a opção 2, pode ser benéfico se estivermos usando várias instruções de depuração em um único método ou classe, em que a estrutura não precisa chamar o isDebugEnabled()método várias vezes (em cada chamada); nesse caso, ele chama o isDebugEnabled()método apenas uma vez e, se o Log4J estiver configurado para o nível de depuração, na verdade, ele chama o isDebugEnabled()método duas vezes:

  1. No caso de atribuir valor à variável debugEnabled, e
  2. Realmente chamado pelo método logger.debug ().

Eu não acho que se escrevermos várias logger.debug()instruções no método ou classe e chamar o debug()método de acordo com a opção 1, isso será uma sobrecarga para o framework Log4J em comparação com a opção 2. Como isDebugEnabled()é um método muito pequeno (em termos de código), ele pode ser um bom candidato para inlining.

Respostas:


248

Nesse caso em particular, a opção 1 é melhor.

A declaração de guarda (checando isDebugEnabled() ) existe para impedir o cálculo potencialmente caro da mensagem de log quando envolve a invocação dos toString()métodos de vários objetos e a concatenação dos resultados.

No exemplo dado, a mensagem de log é uma cadeia constante, portanto, deixar o logger descartá-la é tão eficiente quanto verificar se o logger está ativado e diminui a complexidade do código porque há menos ramificações.

Melhor ainda é usar uma estrutura de log mais atualizada, na qual as instruções de log adotam uma especificação de formato e uma lista de argumentos a serem substituídos pelo criador de logs - mas "preguiçosamente", apenas se o criador de logs estiver ativado. Esta é a abordagem adotada pelo slf4j .

Veja minha resposta a uma pergunta relacionada para obter mais informações e um exemplo de como fazer isso com o log4j.


3
log5j estende log4j em muito da mesma forma que slf4j
Bill Michell

Essa também é a abordagem do java.util.Logging.
Paul

@ Geek É mais eficiente quando o evento de log está desativado porque o nível do log está alto. Consulte a seção "A necessidade de registro condicional" na minha resposta aqui.
24513 Erickson

1
Isso mudou no log4j 2?
SnakeDoc

3
@SnakeDoc Não. É fundamental a chamada de método: as expressões nas listas de métodos são efetivamente avaliadas antes da chamada. Se essas expressões são a) consideradas caras eb) desejadas apenas sob determinadas condições (por exemplo, quando a depuração está ativada), sua única opção é colocar uma verificação de condição em torno da chamada, e a estrutura não pode fazer isso por você. O que ocorre com os métodos de log baseados no formatador é que você pode passar alguns Objetos (que são essencialmente gratuitos) e o criador de logs chamará toString()apenas se necessário.
SusanW

31

Como na opção 1 a cadeia de mensagens é uma constante, não há absolutamente nenhum ganho em agrupar a instrução de log com uma condição; pelo contrário, se a instrução de log estiver ativada para depuração, você avaliará duas vezes, uma vez no isDebugEnabled()método e outra em debug()método. O custo da chamada isDebugEnabled()é da ordem de 5 a 30 nanossegundos, o que deve ser insignificante para os propósitos mais práticos. Portanto, a opção 2 não é desejável, pois polui seu código e não oferece outros ganhos.


17

O uso de isDebugEnabled()é reservado para quando você estiver criando mensagens de log concatenando Strings:

Var myVar = new MyVar();
log.debug("My var is " + myVar + ", value:" + myVar.someCall());

No entanto, no seu exemplo, não há ganho de velocidade, pois você está apenas registrando uma String e não executando operações como concatenação. Portanto, você está apenas adicionando inchaço ao seu código e dificultando a leitura.

Pessoalmente, uso as chamadas no formato Java 1.5 na classe String, desta forma:

Var myVar = new MyVar();
log.debug(String.format("My var is '%s', value: '%s'", myVar, myVar.someCall()));

Duvido que haja muita otimização, mas é mais fácil de ler.

Observe, porém, que a maioria das APIs de log oferece formatação assim: o slf4j, por exemplo, fornece o seguinte:

logger.debug("My var is {}", myVar);

o que é ainda mais fácil de ler.


8
Seu uso de String.format (...), embora facilite a leitura da linha de log, pode realmente afetar o desempenho de uma maneira ruim. A maneira SLF4J de fazer isso envia os parâmetros para o método logger.debug e a avaliação de isDebugEnabled é feita antes da construção da sequência. Do jeito que você está fazendo isso, com String.format (...), a string será construída antes da conclusão da chamada do método para logger.debug, para que você pague a penalidade pela construção da string, mesmo que o nível de depuração seja não habilitado. Desculpe pela nit picking, apenas tentando evitar confusão para iniciantes ;-)
StFS

2
String.format é 40 vezes mais lento que concat & slf4j tem limitação de 2 parâmetros Veja números aqui: stackoverflow.com/questions/925423/… Vi muitos gráficos de criação de perfil em que a operação de formato é desperdiçada em instruções de depuração quando o sistema de produção é executando no nível de log de INFO ou ERRO
AztecWarrior_25


8

Versão curta: Você também pode fazer a verificação booleana isDebugEnabled ().

Razões:
1- Se a lógica / string complicada for concat. é adicionado à sua instrução de depuração, você já terá a verificação no local.
2- Você não precisa incluir seletivamente a instrução nas instruções de depuração "complexas". Todas as instruções são incluídas dessa maneira.
3- Chamar log.debug executa o seguinte antes do log:

if(repository.isDisabled(Level.DEBUG_INT))
return;

Isso é basicamente o mesmo que chamar log. ou gato. isDebugEnabled ().

CONTUDO! É isso que os desenvolvedores do log4j pensam (como está no javadoc e você provavelmente deve seguir em frente).

Este é o método

public
  boolean isDebugEnabled() {
     if(repository.isDisabled( Level.DEBUG_INT))
      return false;
    return Level.DEBUG.isGreaterOrEqual(this.getEffectiveLevel());
  }

Este é o javadoc para ele

/**
*  Check whether this category is enabled for the <code>DEBUG</code>
*  Level.
*
*  <p> This function is intended to lessen the computational cost of
*  disabled log debug statements.
*
*  <p> For some <code>cat</code> Category object, when you write,
*  <pre>
*      cat.debug("This is entry number: " + i );
*  </pre>
*
*  <p>You incur the cost constructing the message, concatenatiion in
*  this case, regardless of whether the message is logged or not.
*
*  <p>If you are worried about speed, then you should write
*  <pre>
*    if(cat.isDebugEnabled()) {
*      cat.debug("This is entry number: " + i );
*    }
*  </pre>
*
*  <p>This way you will not incur the cost of parameter
*  construction if debugging is disabled for <code>cat</code>. On
*  the other hand, if the <code>cat</code> is debug enabled, you
*  will incur the cost of evaluating whether the category is debug
*  enabled twice. Once in <code>isDebugEnabled</code> and once in
*  the <code>debug</code>.  This is an insignificant overhead
*  since evaluating a category takes about 1%% of the time it
*  takes to actually log.
*
*  @return boolean - <code>true</code> if this category is debug
*  enabled, <code>false</code> otherwise.
*   */

1
Obrigado por incluir o JavaDoc. Eu sabia que já tinha visto esse conselho em algum lugar antes e estava tentando encontrar uma referência definitiva. Isto é, se não definitivo, pelo menos muito bem informado.
Simon Peter Chappell

7

Como outros usuários mencionaram, usar a declaração de guarda só é realmente útil se a criação da string for uma chamada demorada. Exemplos específicos disso são quando a criação da string acionará um carregamento lento.

Vale ressaltar que esse problema pode ser solucionado com o Simple Logging Facade for Java ou (SLF4J) - http://www.slf4j.org/manual.html . Isso permite chamadas de método como:

logger.debug("Temperature set to {}. Old temperature was {}.", t, oldT);

Isso só converterá os parâmetros passados ​​em strings se a depuração estiver ativada. SLF4J, como o próprio nome sugere, é apenas uma fachada e as chamadas de log podem ser passadas para o log4j.

Você também pode facilmente "rodar o seu próprio" versão disso.

Espero que isto ajude.


6

A opção 2 é melhor.

Por si só, não melhora o desempenho. Mas garante que o desempenho não diminua. Aqui está como.

Normalmente esperamos logger.debug (someString);

Mas geralmente, conforme o aplicativo cresce, muda muitas mãos, especialmente para desenvolvedores iniciantes, você pode ver

logger.debug (str1 + str2 + str3 + str4);

e similar.

Mesmo se o nível do log estiver definido como ERROR ou FATAL, a concatenação de cadeias acontecerá! Se o aplicativo contiver muitas mensagens no nível DEBUG com concatenações de cadeias, certamente será necessário um impacto no desempenho, especialmente no jdk 1.4 ou abaixo. (Não tenho certeza se as versões posteriores do jdk internall executam qualquer stringbuffer.append ()).

É por isso que a opção 2 é segura. Até as concatenações de strings não acontecem.


3

Como @erickson, depende. Se bem me lembro, isDebugEnabledjá é construir no debug()método de Log4j.
Contanto que você não esteja fazendo alguns cálculos caros em suas instruções de depuração, como loop em objetos, faça cálculos e concatene seqüências de caracteres, você está bem na minha opinião.

StringBuilder buffer = new StringBuilder();
for(Object o : myHugeCollection){
  buffer.append(o.getName()).append(":");
  buffer.append(o.getResultFromExpensiveComputation()).append(",");
}
log.debug(buffer.toString());

seria melhor como

if (log.isDebugEnabled(){
  StringBuilder buffer = new StringBuilder();
  for(Object o : myHugeCollection){
    buffer.append(o.getName()).append(":");
    buffer.append(o.getResultFromExpensiveComputation()).append(",");
  }
  log.debug(buffer.toString());
}

3

Para uma única linha , eu uso um ternário dentro da mensagem de log. Dessa maneira, não faço a concatenação:

ej:

logger.debug(str1 + str2 + str3 + str4);

Eu faço:

logger.debug(logger.isDebugEnable()?str1 + str2 + str3 + str4:null);

Mas para várias linhas de código

ej.

for(Message mess:list) {
    logger.debug("mess:" + mess.getText());
}

Eu faço:

if(logger.isDebugEnable()) {
    for(Message mess:list) {
         logger.debug("mess:" + mess.getText());
    }
}

3

Como muitas pessoas provavelmente estão visualizando essa resposta ao procurar pelo log4j2 e quase todas as respostas atuais não consideram o log4j2 ou alterações recentes, esperamos que isso responda à pergunta.

O log4j2 suporta os fornecedores (atualmente sua própria implementação, mas de acordo com a documentação, está planejado usar a interface de fornecedor do Java na versão 3.0). Você pode ler um pouco mais sobre isso no manual . Isso permite que você coloque a criação cara de mensagens de log em um fornecedor, que só cria a mensagem se for registrada:

LogManager.getLogger().debug(() -> createExpensiveLogMessage());

2

Melhora a velocidade porque é comum concatenar seqüências de caracteres no texto de depuração, o que é caro, por exemplo:

boolean debugEnabled = logger.isDebugEnabled();
if (debugEnabled) {
    logger.debug("some debug text" + someState);
}

1
Se estivermos usando o jdk 1.5 em diante, acho que concatenar seqüências de caracteres não fará diferença.
Silencioso Guerreiro

Por quê? O que o JDK5 faria de diferente?
javashlook

1
No jdk 1.5, se concatenarmos as strings em uma única instrução, internamente, ele usará apenas o método StringBuffer.append (). Portanto, isso não afeta o desempenho.
9990 Silent Warrior

2
A concatenação de strings, sem dúvida, leva tempo. No entanto, não tenho certeza se o descreveria como 'caro'. Quanto tempo economiza no exemplo acima? Comparado com o que o código circundante está realmente fazendo? (por exemplo, leituras de banco de dados ou computação na memória). Eu acho que este tipo de declarações precisa ser qualificado
Brian Agnew

1
Mesmo o JDK 1.4 não criará novos objetos String com concatenação de string simples. A penalidade de desempenho vem do uso de StringBuffer.append () quando nenhuma string deve ser exibida.
Javashlook #

1

Desde a versão Log4j2.4 (ou slf4j-api 2.0.0-alpha1), é muito melhor usar API fluente (ou suporte Java 8 lambda para log lento ), suporte Supplier<?>para argumento de mensagem de log, que pode ser fornecido pelo lambda :

log.debug("Debug message with expensive data : {}", 
           () -> doExpensiveCalculation());

OU com a API slf4j:

log.atDebug()
            .addArgument(() -> doExpensiveCalculation())
            .log("Debug message with expensive data : {}");

0

Se você usa a opção 2, está fazendo uma verificação booleana que é rápida. Na opção um, você está fazendo uma chamada de método (pressionando coisas na pilha) e depois fazendo uma verificação booleana que ainda é rápida. O problema que vejo é consistência. Se algumas de suas instruções de depuração e informação estão agrupadas e outras não, não é um estilo de código consistente. Além disso, alguém mais tarde poderia alterar a instrução de depuração para incluir seqüências de caracteres concatenadas, o que ainda é bastante rápido. Descobri que, quando agrupamos as instruções de depuração e informações em um aplicativo grande e criamos um perfil, economizamos alguns pontos percentuais no desempenho. Não muito, mas o suficiente para fazer valer o trabalho. Agora, tenho algumas configurações de macros no IntelliJ para gerar automaticamente instruções de depuração e informações agrupadas para mim.


0

Eu recomendaria usar a Opção 2 como de fato para a maioria, pois não é muito caro.

Caso 1: log.debug ("uma sequência")

Case2: log.debug ("uma cadeia" + "duas cadeias" + object.toString + object2.toString)

No momento em que um desses itens é chamado, a sequência de parâmetros no log.debug (seja CASE 1 ou Case2) TEM QUE ser avaliada. É isso que todo mundo quer dizer com 'caro'. Se você tiver uma condição anterior, 'isDebugEnabled ()', não será necessário avaliar qual é o local onde o desempenho é salvo.


0

A partir do 2.x, o Apache Log4j possui essa verificação incorporada, portanto, isDebugEnabled()não é mais necessário. Basta fazer um debug()e as mensagens serão suprimidas se não estiverem ativadas.


-1

O Log4j2 permite formatar parâmetros em um modelo de mensagem, semelhante a String.format(), eliminando assim a necessidade isDebugEnabled().

Logger log = LogManager.getFormatterLogger(getClass());
log.debug("Some message [myField=%s]", myField);

Amostra log4j2.properties simples:

filter.threshold.type = ThresholdFilter
filter.threshold.level = debug
appender.console.type = Console
appender.console.name = STDOUT
appender.console.layout.type = PatternLayout
appender.console.layout.pattern = %d %-5p: %c - %m%n
appender.console.filter.threshold.type = ThresholdFilter
appender.console.filter.threshold.level = debug
rootLogger.level = info
rootLogger.appenderRef.stdout.ref = STDOUT
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.